





出 版 者 的 话 


文艺 复兴 以 降 ， 源 远 流 长 的 科学 精神 和 逐步 形成 的 学 术 规范 ， 使 西方 国家 在 自然 科学 的 
各 个 领域 取得 了 仇 断 性 的 优势 ， 也 正 是 这 样 的 传统 ， 使 美国 在 信息 技术 发 展 的 六 十 多 年 间 名 
家 辈出 、 独 领 风骚 。 在 商业 化 的 进程 中 ， 美 国 的 产业 界 与 教育 界 越 来 越 紧密 地 结合 ， 计 算 机 
学 科 中 的 许多 泰山 北斗 同时 身 处 科研 和 教学 的 最 前 线 ， 由 此 而 产生 的 经 典 科学 著作 ， 不 仅 壁 
划 了 研究 的 范畴 ， 还 揭 更 了 学 术 的 源 变 ， 既 遵循 学 术 规 范 ， 又 自 有 学 者 个 性 ， 其 价值 并 不 会 
因 年 月 的 流逝 而 减退 。 

近年 ， 在 全 球 信息 化 大 潮 的 推动 下 ， 我 国 的 计算 机 产业 发 展 迅猛 ， 对 专业 人 才 的 需求 日 
益 迫 切 。 这 对 计算 机 教育 界 和 出 版 界 都 既是 机 遇 ， 也 是 挑战 ， 而 专业 教材 的 建设 在 教育 战略 
上 显得 举足轻重 。 在 我 国信 息 技术 发 展 时 间 较 短 、 从 业 人 员 较 少 的 现状 下 ， 美 国 等 发 达 国 家 
在 其 计算 机 科学 发 展 的 几 十 年 间 积 淀 的 经 典 教材 仍 有 许多 值得 借鉴 之 处 。 因 此 ， 引 进 一 批 国 
外 优秀 计算 机 教材 将 对 我 国 计 算 机 教育 事业 的 发 展 起 积极 的 推动 作用 ， 也 是 与 世界 接轨 、 建 
设 真正 的 世界 一 流 大 学 的 必由之路 。 

机 械 工业 出 版 社 华章 图 文 信息 有 限 公司 较 早 意识 到 “出 版 要 为 教育 服务 "。 自 1998 年 开始 ， 
华章 公司 就 将 工作 重点 放 在 了 进 选 、 移 译 国外 优秀 教材 上 。 经 过 几 年 的 不 懈 努 力 ， 我 们 与 
Prentice Hall, Addison-Wesley, McGraw-Hill, Morgan Kaufmann 等 世界 著名 出 版 公司 建立 了 
和 良好 的 合作 关系 ， 从 它们 现 有 的 数 百 种 教材 中 甄选 出 Tanenbaum ，Stroustrup，Kernighan， 
Jim Gray 等 大 师 名 家 的 一 批 经 典 作品 ， 以 “计算 机 科学 丛书 ”为 总 称 出 版 ， 供 读者 学 习 、 研 
究 及 诬 藏 。 大 理 石 纹理 的 封面 ， 也 正体 现 了 这 套 从 书 的 品位 和 格调 。 

“计算 机 科学 丛书 ”的 出 版 工作 得 到 了 国内 外 学 者 的 鼎力 衷 助 ， 国 内 的 专家 不 仅 提供 了 中 
肯 的 选 题 指导 ， 还 不 辞 劳苦 地 担任 了 翻译 和 审 校 的 工作 ， 而 原 书 的 作者 也 相当 关注 其 作品 在 
中 国 的 传播 ， 有 的 还 专 诚 为 其 书 的 中 译本 作 序 。 迄 今 , “计算 机 科学 丛书 ”已 经 出 版 了 近 百 个 
品种 ， 这 些 书籍 在 读者 中 树立 了 良好 的 口碑 ， 并 被 许多 高 校 采 用 为 正式 教材 和 参考 书籍 ， 为 
进一步 推广 与 发 展 打下 了 坚实 的 基础 。 

随 着 学 科 建 设 的 初步 完善 和 教材 改革 的 逐渐 深化 ， 教 育 界 对 国外 计算 机 教材 的 需求 和 应 
用 都 步 人 一 个 新 的 阶段 。 为 此 ， 华 章 公司 将 加 大 引进 教材 的 力度 ， 在 “华章 教育 ”的 总 规划 
之 下 出 版 三 个 系列 的 计算 机 教材 除 “ 计 算 机 科学 丛书 ”之 外 ， 对 影印 版 的 教材 ， 则 单独 开 
辟 出 “经 典 原版 书库 ”， 同 时 ， 引 进 全 美 通行 的 教学 辅导 书 “Schaum's Outlines” RAAR 
“全 美 经 典 学 习 指 导 系列 "。 为 了 保证 这 三 套 丛 书 的 权威 性 ， 同 时 也 为 了 更 好 地 为 学 校 和 老师 
们 服务 ， 华 章 公司 聘请 了 中 国 科学 院 、 北 京 大 学 、 清 华 大 学 、 国 防 科技 大 学 、 复 旦 大 学 、 上 
海 交通 大 学 、 南 京 大 学 、 浙 江 大 学 、 中 国 科技 大 学 、 哈 尔 滨 工 业 大 学 、 西 安 交通 大 学 、 中 国 
人 民 大 学 、 北 京 航空 航天 大 学 、 北 京 邮电 大 学 、 中 山大 学 、 解 放 军 理工 大 学 、 郑 州 大 学 、 湖 
北 工学 院 、 中 国 国家 信息 安全 测评 认证 中 心 等 国内 重点 大 学 和 科研 机 构 在 计算 机 的 各 个 领域 
的 著名 学 者 组 成 “专家 指导 委员 会 "， 为 我 们 提供 选 题 意见 和 出 版 监督 。 

这 三 套 从 书 是 响应 教育 部 提出 的 使 用 外 版 教材 的 号 召 ， 为 国内 高 校 的 计算 机 及 相关 专业 
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的 教学 度 身 订 造 的 。 其 中 许多 教材 均 已 为 M. I. T., Stanford, U.C. Berkeley, C. M. U. 等 世界 
名 牌 大 学 所 采用 。 不 仅 涵盖 了 程序 设计 、 数 据 结构 、 操 作 系统 、 计 算 机 体系 结构 、 数 据 库 、 
编译 原理 、 软 件 工程 、 图 形 学 、 通 信 与 网 络 、 离 散 数学 等 国内 大 学 计算 机 专业 普遍 开设 的 核 
心 课程 ， 而 且 各 具 特 色 一 一 有 的 出 自 语言 设计 者 之 手 、 有 的 历经 三 十 年 而 不 豪 、 有 的 已 被 全 
世界 的 几 百 所 高 校 采 用 。 在 这 些 圆 熟 通 博 的 名 师 大 作 的 指引 之 下 ， 读 者 必 将 在 计算 机 科学 的 
宫殿 中 由 登 堂 而 信 室 。 

权威 的 作者 、 经 典 的 教材 、 一 流 的 译 者 、 严 格 的 审 校 、 精 细 的 编辑 ， 这 些 因 素 使 我 们 的 
图 书 有 了 质量 的 保证 ， 但 我 们 的 目标 是 尽善尽美 ， 而 反馈 的 意见 正 是 我 们 达到 这 一 终极 目标 
的 重要 帮助 。 教 材 的 出 版 只 是 我 们 的 后 续 服务 的 起 点 。 华 章 公司 欢迎 老师 和 读者 对 我 们 的 工 
作 提 出 建议 或 给 予 指正 ， 我 们 的 联系 方法 如 下 : 


电子 邮件 ， hzedu@hzbook.com 

联系 电话 : (010) 68995264 

联系 地 址 : 北京 市 西城 区 百 万 庄 南 街 1 号 
邮政 编码 ;100037 


专家 指导 委员 会 
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读者 评论 
每 个 Java 程 序 员 都 应 该 反复 研读 《Think in Java》， 并 且 随 身 携带 以 便 随时 参考 。 书 中 的 练习 
颇具 挑战 性 ， 而 有 关 集 合 的 章节 已 至 化 境 ! 本 书 不 仅 帮助 我 通过 了 Sun Certified Java Programmer 


考试 ， 而 且 它 还 是 我 遇 到 Java 问 题 时 ， 求 助 的 首选 书籍 。 
Jim Pleger, Loudoungs. (弗吉尼亚 ) 政府 


这 本 书 比 我 见 过 的 所 有 Java 书 都 要 好 得 多 。 循 序 渐进 …… 非 常 完整 ， 并 搭配 恰到好处 的 范 
例 ， 害 智 而 不 呆板 的 解说 …… 这 使 本 书 的 品质 比 别 的 书 “ 超 出 了 一 个 数量 级 ”"。 与 其 他 Java 书 相 
比 ， 我 发 现 本 书 考虑 非常 周全 、 前 后 一 致 、 理 性 坦诚 、 文 笔 流畅 、 用 词 准确 。 恕 我 直言 ， 这 是 
一 本 学 习 Java 的 理想 书籍 。 

Anatoly Vorobey, 以 色 列 海 法 Technion 大 学 


在 我 所 见 过 的 程序 设计 指南 中 (无 论 何 种 语言 )， 这 绝对 是 最 好 的 一 本 。 
Joakim Ziegler, FIX 系 统管 理 员 


感谢 您 这 本 精彩 的 、 令 人 愉快 的 Java 书 。 
Dr Gavin Pillay, 登记 员 , 南非 爱德华 八 世 医院 


再 次 感谢 您 这 本 杰出 的 书 。 作 为 一 名 不 用 C 语 言 的 程序 员 ， 我 曾经 感到 (学 习 Java) 步履 维 
艰 ， 但 是 您 的 书 让 我 一 目 了 然 。 能 够 一 开始 就 理解 底层 的 概念 和 原理 ， 而 不 是 通过 反复 试验 来 
自己 建立 概念 模型 ， 真 是 太 棒 了 。 我 希望 能 在 不 久 的 将 来 参加 您 的 讨论 课 。 

Randall R. Hawley, 自动 化 工程 师 , Eli Lilly 公 司 





我 见 过 的 计算 机 著作 中 ， 这 是 最 好 的 一 本 。 
Tom Holland 


这 是 我 读 过 的 编程 语言 书 中 最 棒 的 一 本 …… 有 关 Java 的 书 中 最 棒 的 一 本 。 
Ravindra Pai, Oracle 公司 , SUNOS 产品 线 部 门 


我 见 过 的 最 好 的 Java 书 ! 您 做 了 一 项 了 不 起 的 工作 。 您 的 深度 令 人 赞叹 ， 出 版 的 时 候 ， 我 
一 定 会 购买 一 本 。 我 从 1996 年 10 月 就 开始 学 习 Java， 其 间 也 读 过 好 几 本 这 方面 的 书 ， 但 我 觉得 
您 这 本 才 是 “ 必 读 书 ”。 最 近 几 个 月 ， 我 一 直 集 中 精力 于 一 个 完全 用 Java 开 发 的 产品 。 您 的 书 帮 
我 夯实 了 某 些 不 牢固 的 知识 点 ， 并 拓展 了 我 的 知识 面 。 我 甚至 在 面试 签约 者 时 引用 书 中 的 内 容 ， 
作为 参考 的 依据 。 通 过 问 一 些 我 从 书 中 学 到 的 知识 ， 来 判断 他 们 对 Java 的 理解 程度 (例如 ， 数 
组 与 Vector 的 区 别 )。 您 的 书 真是 伟大 ! 

Steve Wilkinson, 资深 专家 , MCI 电信 公司 

伟大 的 书 。 思 今 为 止 我 见 过 的 最 佳 Java 书 籍 。 
Jeff Sinclair, 软件 工程 师 , Kestral 计算 技术 公司 


感谢 您 的 《Thinking in Java》。 早 就 应 该 有 人 把 仅仅 介绍 语言 的 教程 编写 成 富有 思想 、 分 析 
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透彻 的 入门 指南 ， 而 不 是 局 限于 “ 某 个 公司 ”的 语言 。 我 阅读 过 许多 这 方面 的 书 ,但 只 有 您 和 
Patrick Winston 的 作品 给 我 印象 深刻 。 我 已 经 向 客户 推荐 这 本 书 。 再 次 谢谢 您 。 
Richard Brooks, Java 咨询 顾问 , 达拉斯 Sun 专 业 服务 部 门 


Bruce， 您 的 书 真是 太 棒 了 ! 您 的 讲解 清晰 明确 。 通 过 这 本 迷人 的 书 ， 我 获得 了 大 量 Java 知 
识 。 练 习题 也 同样 令 人 着 迷 ， 它 们 对 巩固 各 章 阐述 的 知识 起 到 了 很 好 的 效果 。 我 期 待 您 的 更 多 
品 。 对 您 的 这 本 著作 致 以 谢意 。 阅 读 了 《Thinking in Java) 之 后 ， 我 的 代码 质量 大 有 改善 。 为 

此 我 要 感激 您 ， 我 相信 ， 维 护 我 的 代码 的 程序 员 同 样 也 会 感激 您 。 
Yvonne Watkins, Discover 技术 公司 


其 他 书籍 只 涵盖 Java 的 WHAT (探讨 语法 和 相关 程序 库 ) ， 或 者 只 包含 Java 的 HOW (实际 的 
程序 范例 )。《Thinking in Java》 则 是 我 知道 的 书籍 中 唯一 对 Java 的 WHY 做 出 讲解 的 一 本 。 为 什 
么 要 这 样 设计 ， 为 什么 它 会 那样 运作 ， 为 什么 有 时 候 会 发 生 问题 ， 为 什么 它 在 某 些 方面 比 C++ 
好 而 某 些 方面 不 会 。 虽 然 它 在 教授 程序 语言 的 WHAT 和 HOW 方面 也 很 成 功 ， 但 《Thinking in 
Java) 更 是 爱 钻研 者 的 首选 Java 书 籍 。 

Robert S. Stephenson 


RES TAA. RRA, BEAL REE. 


Chuck Iverson 


我 要 赞美 您 在 《Thinking in Java》 一 书 上 的 表现 。 正 是 有 了 您 这 样 的 人 ， 才 使 得 因特网 充满 
前 景 ， 而 我 想 感谢 您 的 付出 与 努力 。 真 是 感激 不 尽 。 
Patrick Barrell, Network Officer Mamco, QAF Mfg. Inc. 


我 真 的 非常 感激 您 的 热情 与 您 的 作品 。 我 下 载 了 你 的 在 线 书籍 的 每 一 个 修订 版 本 ， 我 正在 
深入 钻研 语言 ， 并 探索 那些 以 前 从 来 不 敢 磁 的 内 容 (对 于 C#、C++、Python 和 Ruby 也 有 作用 )。 
我 至 少 还 有 15 本 Java 书 (为 了 工作 , 我 还 要 掌握 JavaScript 和 PHP 语 言 ), 并且 订阅 了 《Dr Dobbs), 
《JavaPro》、《JDJ》、《JavaWorld》 等 杂志 。 在 深入 钻研 Java (包括 Java 企 业 版 ) 之 后 ， 我 对 您 的 
书 更 加 尊敬 。 它 的 确 是 一 本 有 思想 的 书 。 我 订阅 了 您 的 邮件 列表 ， 并 希望 有 一 天 ， 我 所 探讨 和 
解决 的 问题 能 被 您 扩展 到 解 题 指导 中 (我 将 购买 解 题 指导 ! )。 同 时 ， 非 常 感激 。 


Joshua Long, www.starbuxman.com 


市 面 上 的 Java 书 籍 ， 大 多 比较 适合 初学 者 。 它 们 大 多 数 也 就 只 具备 基础 内 容 ， 范 例 也 大 同 
小 异 。 在 我 见 过 的 富有 思想 性 并 讲解 高 级 主题 的 书籍 中 ， 您 的 是 最 好 的 。 快 点 出 版 吧 ! …… 鉴 
F (Thinking in Java》 带 给 我 的 深刻 印象 ， 我 也 购买 了 《Thinking in C++), 

George Laframboise, LightWorx 技术 咨询 公司 


关于 您 的 《Thinking in C++) (我 工作 的 时 候 ， 它 总 在 书架 上 占据 最 显眼 的 位 置 ) ， 我 曾经 写 

信 告 诉 过 您 我 对 它 的 喜爱 。 现 在 ， 我 通过 您 的 电子 书 仔细 钻研 Java， 我 还 得 说 “我 喜欢 ! ”本 

书 内 容 广博 ， 讲 解 详细 ， 阅 读 起 来 不 像 是 无 味 的 教科 书 。 您 的 书 中 涵盖 了 Java 开发 工作 中 最 重 
要 、 却 很 少 被 提 及 的 概念 一 “原理 ”。 

Sean Brady 


我 同时 用 Java 和 C++ 进 行 开发 ， 您 的 这 两 本 书 是 我 的 救星 。 如 果 我 被 某 个 问题 难 住 了 ， 我 知 
道 可 以 靠 您 的 书 来 :a) 清楚 地 解释 原因 ，b) 找到 符合 我 所 遇 问 题 的 具体 例子 。 我 还 没 找到 另 一 
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位 能 令 我 如 此 反复 热情 推荐 的 作者 《如 果 有 人 愿意 听 我 推荐 的 话 ) 。 
Josh Asbury, A^3 软件 咨询 公司 , 辛辛那提 , 俄亥俄 
您 的 例子 不 仅 清 楚 ， 而 且 容易 理解 。Java 中 的 许多 重要 细节 您 都 考虑 到 了 ， 这 些 内 容 在 编 
排 较 差 的 Java 文 档 中 很 难 找到 。 您 假设 程序 员 已 经 具有 了 基本 知识 ， 这 就 节约 了 读者 的 时 间 。 
Kai Engert, 德国 Innovative 软件 公司 


我 是 《Thinking in C++》 一 书 的 忠实 书迷 ， 我 已 经 将 它 推荐 给 了 我 的 同事 们 。 当 我 读 完 您 的 
Java 书 籍 电子 版 时 ， 我 觉得 ， 您 总 能 保持 高 水 准 的 写作 水 平 。 感 谢 您 ! 
Peter R. Neuwald 


写 得 非常 好 的 Java 书 …… 我 认为 您 在 此 书 上 取得 了 非常 出 色 的 成 就 。 作 为 芝加哥 地 区 Java 兴 
趣 小 组 的 领导 人 ， 我 已 经 多 次 在 我 们 最 近 的 聚会 中 赞扬 您 的 这 本 书 和 您 的 网 站 。 我 想 将 
(Thinking in Java》 作 为 我 们 每 月 聚会 讨论 的 主要 内 容 。 这 样 我 们 可 以 在 聚会 中 对 书 中 的 章节 进 
行 复习 和 讨论 。 

Mark Ertes 

顺便 提 一 下 ,《Thinking in Java 2nd Edition) 俄语 版 依旧 畅销 。 阅 读 此 书 已 经 与 学 习 Java 成 
为 同义词 ， 真 是 太 好 了 。 

Ivan Porty ((Thinking In Java 2nd Edition》 人 饭 语 版 的 译 者 及 出 版 商 ) 


对 于 您 的 辛勤 工作 ， 我 由 圳 感激 。 您 的 书 是 佳作 ， 我 将 这 本 书 推荐 给 我 们 这 儿 的 使 用 者 和 
博士 班 学 生 。 

Hugues Leroy // Irisa-Inria Rennes France, 

Head of Scientific Computing and Industrial Tranfert 


虽然 我 只 读 了 约 40 页 的 《Thinking in Java》， 却 已 经 发 现 本 书 是 我 所 见 过 的 讲述 最 为 清晰 、 
编排 最 为 合理 的 程序 设计 书籍 …… 作 为 一 名 作者 ， 我 可 能 会 有 些 挑剔 。 我 已 经 订购 了 《Thinking 
in C++》， 人 迫不及待 地 想 钻研 一 番 。 对 于 程序 设计 ， 我 还 算是 新 手 ， 因 此 事 事 都 得 学 习 。 这 不 过 
是 一 篇 向 您 的 绝 佳作 品 致谢 的 简短 书信 。 在 痛苦 地 凯 览 大 多 数 语言 艰 涩 、 内 容 散乱 的 计算 机 书 
籍 (包括 那些 有 着 极 佳 口碑 的 书籍 ) 后 ， 我 对 计算 机 书籍 的 阅读 热情 一 度 消退 。 不 过 ， 现 在 我 

又 重 拾 信心 。 
Glenn Becker, Educational Theatre Association 


感谢 您 提供 了 这 么 一 本 精彩 的 书 。 当 我 遇 到 那些 令 人 困惑 的 Java 和 C++ 问题 时 ， 这 本 书 对 
我 最 终 理解 问题 提供 了 极 大 帮助 。 阅 读 您 的 书 令 人 如 沐 春风 。 
Felix Bizaoui, Twin Oaks Industriés, Louisa, Va. 
对 于 这 部 优秀 的 作品 ， 我 必须 向 您 道贺 。 鉴 于 阅读 《Thinking in C++》 的 经 验 ， 我 决定 读 一 
BE (Thinking in Java》， 而 事实 证 明 它 的 确 未 让 人 失望 。 
Jaco van der Merwe, 南非 DataFusion 系 统 公司 软件 专家 
本 书 无 疑 是 我 所 见 过 的 最 佳 的 Java 书 籍 之 一 。 
EF Pritchard, 英国 剑桥 动画 系统 公司 高 级 软件 工程 师 








、 您 的 书 使 那些 我 曾经 读 过 或 草草 翻 过 的 Java 书 显得 更 加 无 用 、 该 骂 。 
Brett g Porter, Art & Logic 公 司 高 级 程序 员 


我 阅读 您 这 本 书 已 经 一 两 个 星期 了 。 与 以 前 我 曾 读 过 的 Java 书籍 比较 ， 您 的 书 似乎 更 能 给 
我 一 个 绝 佳 的 开始 。 我 已 经 把 此 书 推荐 给 我 的 朋友 们 ， 他 们 对 此 书 也 评价 甚 高。 对 于 您 写 出 的 
这 本 著作 ， 请 接受 我 的 恭喜 。 

Rama Krishna Bhupathi, 加 州 圣何塞 TCSI 公司 软件 工程 师 

只 是 很 想 告诉 您 ， 您 这 本 书 是 多 么 杰出 的 作品 。 我 已 将 它 作为 公司 内 部 Java 工 作 的 主要 参 
考 书 。 我 发 现 目录 的 安排 恰如其分 ， 可 以 很 快 找到 需要 的 章节 。 能 够 看 到 这 么 一 本 既 不 拿 API 炒 
冷 饭 ， 也 不 把 程序 员 当 傻瓜 的 书 ， 真 是 太 棒 了 。 

Grant Sayer, 澳大利亚 Ceedata 系统 私人 有 限 公 司 ，Java 组 件 组 长 


RE! 这 是 一 本 可 读 性 强 、 极 富 深度 的 Java 书 籍 。 市 面 上 已 经 有 太 多 质量 低劣 的 Java 书籍 。 
其 中 虽然 也 有 少数 不 错 的 ， 但 在 看 过 您 的 大 作 之 后 ， 我 认为 它 当 然 是 最 好 的 。 
John Root, 伦敦 社会 安全 局 Web 开 发 人 员 


我 才刚 开始 阅读 《Thinking in Java》。 我 想 它 一 定 相当 不 错 ， 因 为 我 很 喜欢 《Thinking in 
Ctt) (我 以 一 名 熟悉 C++、 同 时 希望 提升 自身 能 力 的 程序 员 身份 来 阅读 这 本 书 ) 。 尽 管 我 不 太 部 
悉 Java， 但 本 书 想必 能 令 我 满意 。 您 是 一 位 伟大 的 作家 。 

Kevin K. Lewis, ObjectSpace 公 司 技术 专家 


我 想 这 是 一 本 了 不 起 的 书 。 我 所 有 的 Java 知 识 都 学 自 这 本 书 。 感 谢 您 让 大 家 可 以 从 Internet 

上 免费 取得 这 本 书 。 如 果 没 有 您 的 付出 ， 我 至 今 恐 怕 仍然 对 Java 一 无 所 知 。 本 书 最 棒 的 一 点 ， 
莫 过 于 它 同 时 也 说 明了 Java 不 好 的 一 面 ， 而 不 像 那些 商业 宣传 资料 。 您 的 表现 真是 优秀 。 

Frederik Fix, 比利时 


我 始终 热 中 读 您 的 著作 。 几 年 以 前 ， 当 我 开始 学 习 C++ 时 ， 是 《C++ Inside & Out》 带 领 我 
进入 C++ 的 迷人 世界 。 那 本 书 帮助 我 得 到 了 更 好 的 机 会 。 现 在 ， 为 了 更 进一步 钻研 知识 ， 我 兴 
起 了 学 习 Java 的 念头 ,我 无 意 中 又 碰见 了 《Thinking in Java》。 毫 无 疑问 ， 我 认为 自己 不 再 需要 
其 他 书籍 。 它 是 那么 的 令 人 难以 置信 。 阅 读 此 书 的 过 程 ， 就 像 重新 发 据 自 我 一 样 。 我 学 习 Java 
至 今 只 有 一 个 月 ， 现 在 对 Java 的 体会 日 益 加 深 ， 这 一 切 都 不 得 不 由 圳 感谢 您 。 

Anand Kumar S., 印度 Computervision 公 司 软件 工程 师 


您 的 书 作为 综合 性 的 导论 ;是 如 此 出 色 。 
Peter Robinson, 剑桥 大 学 计算 机 实验 室 


在 帮助 我 学 习 Java 的 书籍 中 ， 这 一 本 显然 是 最 好 的 。 我 只 是 想 让 您 知道 ， 我 觉得 自己 能 够 
读 到 这 本 书 是 多 么 幸运 。 谢 谢 ! 
Chuck Peterson, [VIS 国际 公司 Intermnet 产 品 线 产品 组 长 
了 不 起 的 一 本 书 。 自 从 我 开始 学 习 Java， 这 已 经 是 第 三 本 了 。 目 前 我 大 概 阅读 了 三 分 之 二 ， 
并 打算 把 它 读 完 。 我 能 够 找到 这 本 书 ， 是 因为 这 本 书 被 用 于 Lucent 技术 公司 的 某 些 内 部 课程 ， 
而 且 有 个 朋友 告诉 我 这 本 书 可 以 在 网 络 上 找到 。 很 棒 的 作品 。 
Jerry Nowlin, Lucent 技术 公司 MTS 部 门 





在 我 所 读 过 的 六 本 Java 书 籍 中 ， 您 的 《Thinking in Java》 显 然 最 好 ， 也 最 清晰 易 懂 。 
Michael Van Waas 博 士 , TMR Associates 公 司 总 裁 


感谢 您 的 《Thinking in Java》。 您 的 作品 真是 精彩 ! 更 不 必 说 它 可 以 免费 从 网 络 下 载 了 ! 作 
为 一 名 学 生 ， 我 觉得 您 的 书籍 是 无 价 之 宝 (我 也 有 一 本 《C++ Inside & Out》， 它 同样 是 一 本 伟 
大 的 C++ 书籍 )， 因 为 您 的 书 不 仅 教导 我 应 该 怎么 做 ， 也 教导 我 这 么 做 的 原因 所 在 ， 这 一 点 对 
C++ 或 Java 学 习 者 建立 起 坚固 基础 非常 重要 。 我 有 许多 和 我 一 样 喜 爱 程序 设计 的 朋友 ， 我 也 对 他 
们 提起 您 的 书 。 他 们 觉得 真是 太 棒 了 ! 再 次 谢谢 您 ! 顺道 一 提 ， 我 是 印度 尼 西亚 人 ， 就 住 在 
“JINE” (Java), 

Ray Frederick Djajadinata, 印度 尼 西 亚 牙 加 达 Trisakti 大 学 学 生 


单 是 将 作品 免费 放 在 网 络 上 这 种 气度 ， 就 令 我 震惊 不 已 。 我 想 ， 我 应 该 让 您 知道 ， 对 于 您 
的 工作 ， 我 是 多 么 感激 与 尊敬 。 
Shane LeBouthillier, 加 拿 大 Alberta 大 学 计算 机 工程 系 学 生 


我 得 告诉 您 ， 每 个 月 我 都 在 期 待 您 的 专栏 。 作 为 面向 对 象 程序 设计 领域 的 新 手 ， 我 要 感谢 

您 花 在 那些 基础 主题 上 的 时 间 和 思考 。 我 已 经 下 载 了 您 的 这 本 书 ， 而 且 我 一 定 会 在 本 书 出 版 的 
时候 购买 一 本 。 感 谢 您 对 我 的 帮助 。 

Dan Cashmer, B. C. Ziegler 公 司 


能 够 完成 这 么 了 不 起 的 作品 ， 准 喜 您 。 开 始 ， 我 偶然 发 现 了 《Thinking in Java》 的 PDF 版 本 。 
甚至 在 我 读 完 之 前 ， 我 又 跑 到 书店 找到 了 (Thinking in C++》。 我 已 经 在 计算 机 领域 工作 了 八 年 
多 ， 做 过 顾问 、 软 件 工程 师 、 教 师 /教练 ， 最 近 则 从 事 自由 职业 。 所 以 我 觉得 自己 也 算是 见 多 识 
广 了 (注意 ， 不 是 “无 所 不 知 "， 而 只 是 “ 见 多 识 广 )。 不 过 ， 这 些 书 使 得 我 的 女 朋友 称 我 为 
“果子 "。 我 并 不 反对 ， 只 不 过 我 发 现 自己 已 经 远 远 超过 这 个 阶段 。 我 发 现 自己 如 此 喜爱 这 两 本 
书 ， 我 以 前 接触 过 或 购买 过 的 其 他 计算 机 书籍 ， 都 无 法 与 之 相 比 。 这 两 本 书 都 有 极 佳 的 写作 风 
格 ， 对 于 每 个 新 主题 都 有 很 好 的 介绍 ， 书 中 充满 了 罕 智 的 见解 。 干 得 好 。 


Simon Goland, simonsez@smartt.com, Simon Says 咨询 公司 
我 得 说 ， 您 的 《Thinking in Java》 真 是 了 不 起 。 它 正 是 我 要 找 的 那 种 书 。 尤 其 那些 讨论 优秀 
与 拙劣 的 Java 软件 设计 的 章节 ， 完 全 就 是 我 想 要 的 。 
Dirk Duehr 德国 贝塔 斯 受 集团 Lexikon 公司 
感谢 您 写 了 两 本 著作 :《Thinking in C++) #1 (Thinking in Java》。 在 面向 对 象 程序 设计 的 学 
习 过 程 中 ， 您 带 给 我 巨大 帮助 。 
Donald Lawson, DCL Enterprises 公 司 
感谢 您 花 时 间 来 撰写 这 么 一 本 很 有 用 的 Java 书 籍 。 如 果 是 教学 让 您 明白 了 某 些 事情 的 话 ， 
到 如 今 您 一 定 极为 满意 自己 的 成 就 。 
Dominic Turner, GEAC Support 
我 曾 读 过 的 最 棒 的 Java 书籍 -一 我 真 的 读 过 不 少 。 
Jean-Yves MENGANT, 法 国 巴 黎 NAT-SYSTEM 公 司 首席 条 统 架构 师 








(Thinking in Java》 涵 盖 全 面 ， 讲 解 清晰 。 本 书 极 易 阅读 ， 而 且 程 序 代 码 也 是 如 此 。 
Ron Chan 博 士 , WE #4 Expert Choice 公 司 


您 的 书 真 好 。 我 读 过 许多 程序 设计 书籍 ， 但 是 这 本 书 中 您 对 程序 设计 的 深刻 见解 依然 深 深 
触动 了 我 。 
Ningjian Wang, Vanguard 集团 信息 系统 工程 师 
(Thinking in Java》 是 一 本 既 优秀 ， 又 容易 阅读 的 书籍 。 我 向 所 有 的 学 生 推荐 它 。 
Dr Paul Gorman, 新 西 兰 Otago 大 学 计算 机 科学 系 
依靠 您 的 书 ， 我 现在 已 经 理解 了 面向 对 象 程序 设计 的 含义 …… 我 相信 ，Java 比 Perl 更 直接 ， 
甚至 更 容易 。 
Torsten Römer, Orange ft & 2 4) 
您 打破 了 “天 下 没有 白 吃 的 午餐 ”这 名 话语。 不 是 那 种 施舍 性 质 的 午餐 ， 而 是 连 美食 家 都 
觉得 美味 的 午餐 。 他 们 都 会 为 此 感激 您 。 
Jose Suriol, Scylax 公司 
感谢 有 机 会 看 到 这 本 书 成 为 一 部 杰作 ! 在 这 个 主题 上 ， 本 书 绝对 是 我 所 读 过 的 最 佳 书籍 。 
Jeff Lapchinsky, Net Results 技术 公司 程序 员 
您 的 书简 明 扼要 ， 容 易 理解 ， 而 且 读 起 来 充满 乐趣 。 
Keith Ritchie, KL 集团 公司 Java 研 发 组 
确实 是 我 所 读 过 的 最 好 的 Java 书籍 ! 
Daniel Eng 
生平 所 见 最 好 的 Java 书籍 ! 
Rich Hoffarth, West 集团 高 级 架构 师 
感谢 您 带 来 了 如 此 精彩 的 一 本 好 书 。 通 读 各 个 章节 带 给 我 极 大 的 乐趣 。 
Fred Trimble, Actium 公司 
SET CORA. GERTLER mA, BRL ict AE 
得 非常 简单 ， 同 时 令 人 愉快 。 感 谢 您 这 本 真正 精彩 的 指南 。 
Rajesh Rau, 软件 顾问 
(Thinking in Java》 撼 动 了 整个 自由 世界 ! 
Miko O'Sullivan, Idocs 公司 总 裁 





KF «Thinking in C++)° 


最 好 的 书 ! 199546 (Software Development 》 杂 志 Jolt 大 奖 得 主 ! 
“本 书 成 就 非凡 。 您 应 该 在 书架 上 也 摆 一 本 。 其 中 讨论 输入 、 输 出 流 的 章节 ， 在 我 所 见 过 的 
有 关 此 主题 的 论著 中 ， 它 是 表述 最 全 面 、 也 最 容易 理解 的 。” 
Al Stevens, (Dr. Dobbs Journal》 的 特约 编辑 


“对 于 如 何 重新 认识 面向 对 象 程序 的 构造 ，Eckel 的 这 本 书 是 唯一 能 做 出 如 此 清晰 解释 的 书 
籍 。 同 时 ， 它 也 是 透彻 讲解 C++ 的 优秀 教程 。” 
Andrew Binstock, (Unix Review》 的 编辑 


“Bruce 对 C++ 的 洞察 力 ， 不 断 令 我 感到 惊讶 。《Thinking in C++》 则 是 他 迄今 为 止 所 有 绝妙 

想法 的 最 佳 合集 。 有 关 C++ 的 种 种 难题 ， 如 果 您 需要 清楚 的 解答 ， 请 买 下 这 本 杰作 。” 
Gary Entsminger 《The Tao of Objects》 的 作者 

(Thinking in C++》 耐 心 而 系统 地 对 C++ 种 种 特性 的 使 用 时 机 与 方式 进行 了 探讨 。 包 括 ; 内 
联 函 数 、 引 用 、 操 作 符 重 载 、 继 承 、 动 态 对 象 。 也 包括 了 许多 高 级 主题 ， 比 如 模板 、 异 常 、 多 
重 继承 的 恰当 用 法 。 对 这 些 交织 在 一 起 ， 最 后 形成 了 Eckel 对 对 象 和 程序 设计 的 独特 看 法 。 它 是 
每 个 C++ 开 发 者 书架 上 的 必 备 好 书 。 如 果 您 正 以 C++ 从 事 严肃 的 开发 工作 ， 那 么 《Thinking in 
C++》 是 您 的 必 备 书籍 之 一 。 


Richard Hale Shaw，《PC Magazine》 的 特约 编辑 


O 本 书 第 1 卷 与 第 2 卷 的 中 文 版 与 英文 版 均 已 由 机 械 工业 出 版 社 出 版 。 一 编辑 注 





译 者 序 


时 隔 两 年 多 ，《Java 编 程 思想 〈 第 4 版 )》 的 中 文 版 又 要 和 广大 Java 程 序 员 和 爱好 者 们 见面 了 。 
这 是 Java 语 言 本 身 不 断 发 展 和 完善 的 必然 要 求 ， 也 是 本 书 作者 Bruce Eckei 和 孜孜不倦 的 创作 激情 和 
灵感 所 结 出 的 硕果 。 

《Java 编 程 思 想 (第 4 版 )》 以 Java 最 新 的 版 本 JDK5.0 为 基础 ， 在 第 3 版 的 基础 上 ， 添 加 了 最 新 
的 语言 特性 ， 并 且 对 第 3 版 的 结构 进行 了 调整 ， 使 得 所 有 章节 的 安排 更 加 遵照 循序 渐进 的 特点 ， 
同时 每 一 章 的 内 容 在 分 量 上 也 都 更 加 均衡 ， 这 使 读者 能 够 更 加 容易 地 阅读 本 书 并 充分 了 解 每 章 
所 讲述 的 内 容 。 在 这 里 我 们 再 次 向 Bruce Eckel 致 敬 ， 他 不 但 向 我 们 展示 了 什么 样 的 书籍 才 是 经 
典 书籍 ， 而 且 还 展示 了 经 典 书籍 怎样 才能 精益 求 精 ， 长 盛 不 衰 。 

Java 已 经 成 为 了 编程 语言 的 骄子 。 我 们 可 以 看 到 ， 越 来 越 多 的 大 学 在 教授 数据 结构 、 程 序 
设计 和 算法 分 析 等 课程 时 ， 选 择 以 Java 语 言 为 载体 。 这 说 明 Java 语 言 已 经 是 人 们 构建 软件 系统 时 
主要 使 用 的 一 种 编程 语言 。 但 是 ， 黎 握 好 Java 语 言 并 不 是 一 件 可 以 轻松 完成 的 任务 ， 如 何 真 正 
掌握 Java 语 言 ， 从 而 编写 出 健壮 的 、 高 效 的 以 及 灵活 的 程序 是 Java 程 序 员 们 面临 的 重大 挑战 。 

《Java 编 程 思想 (第 4 版 )》 就 是 一 本 能 够 让 Java 程 序 员 轻松 面 对 这 一 挑战 ， 并 最 终 取得 胜利 
的 经 典 书籍 。 本 书 深 入 浅 出 、 循 序 渐进 地 把 我 们 领 和 人 Java 的 世界 ， 让 我 们 在 不 知 不 觉 中 就 学 会 
了 用 Java 的 思想 去 考虑 问题 、 解 决 问题 。 本 书 不 仅 适合 Java 的 初学 者 ， 更 适合 于 有 经 验 的 Java 程 
序 员 ， 这 正 是 本 书 的 魅力 所 在 。 但 是 ， 书 中 并 没有 涵盖 Java 所 有 的 类 、 接 口 和 方法 ， 因 此 ， 如 
果 你 希望 将 它 当 作 Java 的 字典 来 使 用 ， 那 么 显然 就 要 失望 了 。 

我 们 在 翻译 本 书 的 过 程 中 力求 忠于 原著 ， 为 了 保持 连贯 性 ， 对 原 书 第 3 版 中 仍然 保持 不 变 的 
部 分 ， 我 们 对 译文 除了 个 别 地 方 之 外 ， 也 没 做 修改 。 对 于 本 书 中 出 现 的 大 量 的 专业 术语 尽量 遵 
循 标准 的 译 法 ， 并 在 有 可 能 引起 歧义 之 处 注 有 英文 原文 ， 以 方便 读者 对 照 与 理解 。 

全 书 由 陈 吴 哨 翻 译 ， 郭 嘉 也 参与 了 部 分 翻译 工作 。 由 于 水 平 有 限 ， 书 中 出 现 错误 与 不 妥 之 
处 在 所 难免 ， 奶 请 读者 批评 指正 。 
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一 开始 ， 我 只 是 将 Java 看 作 “ 又 一 种 程序 设计 语言 。 从 许多 方面 看 ， 它 也 的 确 如 此 。 

但 随 着 时 间 流 逝 ， 以 及 对 Java 的 深入 研究 ， 我 渐渐 发 现 ， 与 我 所 见 过 的 其 他 编程 语言 相 比 ， 
Java 有 着 完全 不 同 的 核心 目的 。 

程序 设计 其 实 是 对 复杂 性 的 管理 : 待 解决 问题 的 复杂 性 ， 以 及 用 来 解决 该 问题 的 工具 的 复 
杂 性 。 正 是 这 种 复杂 性 ， 导 致 多 数 程序 设计 项 目 失败 。 在 我 所 知 的 所 有 程序 设计 语言 中 ， 几 乎 
没有 哪个 将 自己 的 设计 目标 专注 于 克服 开发 与 维护 程序 的 复杂 性 9 。 当 然 ， 有 些 编程 语言 在 设计 
决策 时 也 曾 考虑 到 复杂 性 的 问题 ， 然 而 ， 总 是 会 有 其 他 议题 被 认为 更 有 必要 加 入 到 该 语言 中 。 
于 是 不 可 避免 地 ， 正 是 这 些 所 谓 更 必要 的 议题 导致 程序 员 最 终 “ 头 擅 南 墙 "。 例 如 ，C++ 选 择 向 
后 兼容 C (以 便 更 容易 吸引 C 程 序 员 ) ， 以 及 具备 C 一 样 的 高 效率 。 这 两 点 都 是 非常 有 益 的 设计 目 
标 ， 也 确实 促成 了 C++ 的 成 功 ， 然 而 它们 却 暴 露出 更 多 的 复杂 性 问题 ， 而 这 也 使 得 很 多 项 目 不 得 
善终 (你 自然 可 以 责怪 程序 员 或 者 项 目 管理 ， 但 是 ， 如 果 一 种 语言 能 够 帮助 你 解决 错误 ， 那 何 
乐 而 不 为 呢 ? )。 再 看 一 个 例子 ，Visual Basic (VB) 选择 与 Basic 绑 在 一 起 ， 而 Basic 并 未 被 设计 
为 具备 可 扩展 性 的 程序 设计 语言 ， 结 果 呢 ， 建 立 在 VB 之 上 的 所 有 扩展 都 导致 了 无 法 维护 的 语法 。 
还 有 Perl， 它 向 后 兼容 awk、sed、grep， 以 及 所 有 它 打算 替代 的 Unix 工 具 ， 结 果 呢 ， 人 们 开始 指 
资 Perl 程序 成 了 “不 可 阅读 (write-only) 的 代码 ”( 即 ， 只 要 稍 过 一 会 儿 ， 你 就 读 不 懂 刚 完成 的 
程序 了 )。 从 另 一 个 角度 看 ， 在 设计 C++、VB、Perl 以 及 Smalltalk 之 类 的 程序 设计 语言 时 ， 设 计 
师 也 都 为 解决 复杂 性 问题 做 了 某 种 程度 的 工作 。 并 且 ， 正 是 解决 某 类 特定 问题 的 能 力 ， 成 就 了 
它们 的 成 功 。 

随 着 对 Java 的 了 解 越 来 越 深 ，Sun 对 Java 的 设计 目标 给 我 留 下 了 最 深刻 印象 ， 那 就 是 ;为 程 
序 员 减 少 复杂 性 。 用 他 们 的 话说 就 是 :“ 我 们 关心 的 是 , 减少 开发 健壮 代码 所 需 的 时 间 以 及 困难 。” 
在 早期 ， 这 个 目标 使 得 代码 的 运行 并 不 快 (Java 程 序 的 运行 效率 已 经 改善 了 ) ， 但 它 确实 显 落地 
缩短 了 代码 的 开发 时 间 。 与 用 C++ 开发 相同 的 程序 相 比 ， 采 用 Java 只 需 一 半 甚 至 更 少 的 开发 时 间 。 
仅 此 一 项 ， 就 已 经 能 节约 无 法 估量 的 时 间 与 金钱 了 。 然 而 Java 并 未 止步 于 此 。 它 开始 着 手 解决 日 
渐变 得 重要 的 各 种 复杂 任务 ， 例 如 多 线程 与 网 络 编程 ， 并 将 其 作为 语言 特性 或 以 工具 库 的 形式 
纳入 Java， 这 使 得 开发 此 类 应 用 变 得 倍加 简单 。 最 终 ，Java 解 决 了 一 些 相当 大 的 复杂 性 问题 ， 跨 
平台 编程 、 动 态 代码 修改 ， 甚 至 是 安全 的 议题 。 它 让 你 在 面 对 其 中 任何 一 个 问题 时 ， 都 能 从 
“举步维艰 ”到 “起 立 鼓掌 ”。 抛 去 我 们 都 能 看 到 的 性 能 问题 ，Java 确 实 非常 精彩 地 履行 了 它 的 诺 
言 : 极 大 地 提升 程序 员 的 生产 率 。 

同时 ，Java 正 从 各 个 方面 提升 人 们 相互 通讯 的 带宽 。 它 使 得 一 切 都 变 得 更 容易 :编写 程序 ， 
团队 合作 ,创建 与 用 户 交 户 的 用 户 界面 ， 在 不 同类 型 的 机 器 上 运行 程序 ， 以 及 编写 通过 因特网 
通信 的 程序 。 

我 认为 ， 通 讯 变革 的 成 果 并 不 见得 就 是 传输 巨 量 的 比特 。 我 们 所 看 到 的 真正 变革 是 人 与 人 


O 不 过 ,我 相信 Python 语言 非常 接近 该 目标 了 。 参 见 wwwpython org。 
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之 间 的 通讯 变 得 更 容易 了 : 无 论 是 一 对 一 的 通信 ， 还 是 群体 与 群体 之 间 ， 甚 至 整个 星球 之 间 的 
通信 。 我 曾 听闻 ， 在 足够 多 的 人 之 间 的 相互 联系 之 上 ， 下 一 次 变革 将 是 一 种 全 球 意识 的 形成 。 
Java 说 不 定 就 是 促进 该 变革 的 工具 ， 至 少 ， 它 所 具备 的 可 能 性 使 我 觉得 ， 教 授 这 门 语言 是 非常 有 
意义 的 一 件 事情 。 


Java SE5 与 SE6 


本 书 的 第 4 版 得 益 于 Java 语 言 的 升级 。Sun 起 初 称 其 为 JDK1.5， 稍 后 改作 JDK5 或 J2SE5， 最 终 
Sun 弃 用 了 过 时 的 “2"， 将 其 改 为 Java SES, Java SE5 的 许多 变化 都 是 为 了 改善 程序 员 的 体验 。 你 
将 会 看 到 ，Java 语 言 的 设计 者 们 并 未 完全 成 功 地 完成 该 任务 ， 不 过 ， 总 的 来 说 ， 他 们 已 经 向 正确 
的 方向 迈 出 了 一 大 步 。 

新 版 的 一 个 重要 目标 就 是 完整 地 吸收 Java SE5/6 的 改进 ， 并 通过 本 书 介绍 以 及 应 用 这 些 变化 。 
这 意味 着 本 书 基 本 可 以 称 之 为 “只 限 Java SE5/6"。 并 且 ， 书 中 的 多 数 代码 并 没有 经 过 老 版 本 的 
Java 编 译 测试 ， 所 以 如 果 你 使 用 的 是 老 版 本 的 Java， 编 译 可 能 会 报错 并 中 止 。 不 过 ， 我 觉得 这 样 
FIKF 

如 果 你 不 得 不 采用 老 版 本 的 Java， 我 仍然 为 你 在 www.MindView.net 提 供 了 本 书 早期 版 本 的 免 
费 下 载 。 基 于 某 些 原因 ， 我 决定 不 提供 本 书 当前 版 本 的 免费 电子 版 。 

Java SE6 

本 书 是 一 个 非常 耗 时 的 ， 且 具有 里 程 碑 意义 的 一 个 项 目 。 就 在 本 书 出 版 之 前 ，Java SEO ( 代 
号 野马 mustang) 已 经 发 布 了 beta 版 。 虽 然 Java SE6 中 的 一 些小 变化 ， 对 书 中 的 代码 示例 有 一 点 影 
响 , 但 其 主要 的 改进 对 本 书 的 绝 大 部 分 内 容 并 没有 影响 。 因 为 Java SE6 主 要 关注 于 提升 速度 ， 以 
及 改进 一 些 (不 在 本 书 讨论 范围 之 内 ) 类 库 的 特性 。 

本 书 中 代码 全 部 用 Java SE6 的 一 个 发 布 候选 版 (RC) 进行 过 测试 ， 因 此 我 不 认为 Java SE6 正 
式 发 布 时 会 有 什么 变化 能 够 影响 本 书 的 内 容 。 如 果 到 时 真 的 有 什么 重要 的 改变 ， 我 将 更 新 本 书 
中 的 代码 ， 你 可 以 通过 www.MindView.net 下 载 。 

本 书 的 封面 已 经 指出 ， 本 书面 向 “Java SE5/6" 。 也 就 是 说 本 书 的 撰写 “面向 Java SE5 及 其 为 
Java 语 言 引入 的 重大 变化 ， 同 时 也 适用 于 Java SE6”。 
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为 一 本 书写 作 新 版 时 ， 作 者 最 满意 的 是 把 事情 做 得 “恰如其分 "。 这 是 我 从 本 书 上 一 个 版 
本 发 布 以 来 所 学 到 的 东西 。 通 常 而 言 ， 这 种 见识 正如 谚语 所 云 ,“ 学 习 就 是 从 失败 中 汲取 教训 。” 
并 且 ， 我 也 借 机 进行 了 一 些 修订 。 与 往常 一 样 ， 一 个 新 的 版 本 必 将 带 来 引人入胜 的 新 思想 。 此 
时 ， 新 发 现 带 来 的 喜悦 ， 采 用 比 以 往 更 好 的 形式 表达 思想 的 能 力 ， 已 经 远 远 超过 了 可 能 引入 的 
小 错误 。 

这 也 是 对 不 断 在 我 脑 中 盘旋 低语 着 的 一 种 挑战 ， 那 就 是 让 持 有 本 书 老 版 本 的 读者 也 愿意 购 
买 新 的 版 本 。 这 些 促使 着 我 尽 可 能 改进 ， 重 写 ， 以 及 重新 组 织 内 容 ， 为 热忱 的 读者 们 献上 一 本 
全 新 的 ， 值 得 拥有 的 书 。 
改变 

此 版 本 中 将 不 再 包含 以 往 本 书 中 所 携带 的 CD 光盘 。 该 CD 中 的 重要 部 分 《Thinking in C》 的 
多 媒体 教程 (由 Chuck Allison 为 MindView 创 建 )， 现 在 提供 了 可 下 载 的 Flash 版 本 。 该 教程 是 为 不 


[En 
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熟悉 C 语 法 的 读者 所 准备 的 。 虽 然 ， 本 书 用 了 两 章 对 语法 做 了 较为 完整 的 介绍 ， 然 而 对 于 没有 相 
应 背景 知识 的 读者 而 言 ， 这 也 许 仍然 不 够 。 而 《Thinking in C》 正 是 为 了 帮助 这 些 读者 提升 到 必 
要 的 程度 。 

完全 重 写 了 “并 发 ”这 一 章 (以 前 称 为 “多 线程 ") ， 以 符合 Java SE5 并 发 类 库 的 重大 改变 。 
它 将 为 读者 了 解 并 发 的 核心 思想 打下 基础 。 如 果 没有 这 些 核心 的 基础 知识 ， 读 者 很 难 理解 关于 
线程 的 更 复杂 的 议题 。 我 花 了 很 多 个 月 撰写 这 一 章 ， 深 陷 “ 并 发 ”的 地 狱 之 中 ， 最 终 ， 这 一 章 
不 仅 涵盖 了 基础 知识 ， 而 且 大胆 地 引入 了 一 些 高 级 议题 。 

而 对 于 Java SE5 所 具有 的 每 一 个 重大 的 新 特性 ， 本 书 都 有 一 个 新 的 章节 与 之 对 应 。 其 他 的 新 
特性 则 加 入 到 了 原 有 的 章节 中 。 我 还 一 直 在 研究 设计 模式 ， 因 此 在 本 书 中 ， 也 介绍 了 设计 模式 
的 相关 内 容 。 

本 书 经 历 了 重大 的 重组 。 这 大 多 源 自 教授 Java 的 过 程 ， 以 及 我 对 于 “章节 ”的 意义 的 重新 思 
考 。 以 前 ， 我 会 不 假 思索 地 认为 ， 每 个 “章节 ”应 该 包含 一 个 “足够 大 的 ”主题 。 但 是 ， 在 我 
教授 设计 模式 的 时 候 ， 我 发 现 ， 如 果 每 次 只 介绍 一 个 模式 〈 即 使 讲课 的 时 间 很 短 ) ， 然 后 立刻 组 
织 大 家 做 练习 ， 此 时 那些 学 员 们 的 表现 是 最 好 的 (我 发 现 ， 这 种 节奏 对 于 我 这 个 老师 而 言 也 更 
有 乐趣 )。 因 此 ， 在 这 一 版 中 ， 我 试 着 打破 按 主题 划分 章节 的 做 法 ， 也 不 理会 章节 的 长 度 。 我 想 ， 
这 也 是 一 个 改进 。 

我 同样 也 认识 到 代码 测试 的 重要 性 。 必 须要 有 一 个 内 建 的 测试 框架 ， 并 且 每 次 你 开发 系统 
时 都 必须 进行 测试 。 否 则 ， 根 本 没有 办 法 知道 代码 可 靠 与 否 。 为 了 做 到 这 一 点 ， 我 开发 了 一 个 
测试 框架 以 显示 和 验证 本 书 中 每 一 个 程序 的 输出 结果 。( 该 框架 是 用 Python 编写 的 ， 你 可 以 在 
www.MindView.net 找 到 可 下 载 的 代码 。) 关于 测试 的 话题 在 附录 中 有 讨论 ， 你 可 以 在 
http://MindView.net/Books/BetterJava 找 到 。 其 中 还 包含 了 其 他 一 些 基本 技术 ， 我 认为 所 有 程序 员 
都 应 该 将 它们 加 入 到 自己 的 工具 箱 中 。 

此 外 ， 我 还 仔细 检查 了 书 中 的 每 一 个 示例 ， 并 且 问 我 自己 ，“ 我 为 什么 采用 这 种 方式 实现 ?” 
对 大 多 数 的 示例 ， 我 都 做 了 一 定 程度 的 修订 与 改进 ， 使 得 这 些 示例 更 加 贴切 。 同 时 ， 也 传达 出 
我 所 认为 的 Java 编 程 中 的 最 佳 实践 (至 少 起 到 抛砖引玉 的 作用 )。 许 多 以 前 的 示例 都 经 过 了 重新 
设计 与 重新 编写 ,同时 ， 刷 除了 不 再 有 意义 的 示例 ， 也 添加 了 新 的 示例 。 

读者 们 为 此 书 的 前 三 个 版 本 提出 了 许多 许多 精彩 的 意见 。 这 自然 使 我 觉得 非常 高 兴 。 不 过 ， 
偶尔 读者 也 会 有 抱怨 ， 例 如 有 读者 埋 急 “本 书 太 长 了 "。 对 我 而 言 ， 如 果 “ 页 数 太 多 ”是 你 唯一 
的 苦恼 ， 那 这 真 令 人 哭笑不得 。( 据 说 奥地利 皇帝 曾 抱怨 莫扎特 的 音乐 “音符 太 多 ”|! 我 可 不 是 
想 把 自己 比 作 莫扎特 。) 此 外 ， 我 只 能 猜测 ， 发 出 这 种 抱怨 的 读者 还 不 了 解 Java 语 言 的 博大 精深 ， 
而 且 也 没有 看 过 这 一 领域 的 其 他 书籍 。 无 论 如 何 ， 在 这 一 版 中 ， 我 已 经 删 碱 了 过 时 无 用 ， 或 不 
再 重要 的 内 容 。 总 的 来 说 ， 我 已 经 尽 我 所 能 仔细 复查 了 全 书 ， 进 行 了 必要 的 增删 与 改进 。 对 于 
删除 旧 的 章节 ， 我 还 是 挺 放 心 的 。 因 为 原始 的 材料 在 网 站 上 都 有 (www.MindView.net) 。 本 书 从 
第 一 版 到 第 三 版 ， 以 及 本 书 的 附录 ， 都 可 以 从 此 网 站 上 下 载 。 

对 于 仍然 不 能 接受 本 书 篇 幅 的 读者 ， 我 向 你 们 道 菊 。 请 相信 ， 我 已 经 尽 我 所 能 精简 本 书 的 
长 度 了 。 


封面 图 片 的 故事 
(Thinking in Java》 的 封面 创作 灵感 来 自 于 美国 的 Arts & Crafts 运 动 。 该 运动 始 于 世纪 之 交 ， 
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并 在 1900 到 1920 年 间 达 到 顶峰 。 它 起 源 于 英格兰 ， 是 对 工业 革命 带 来 的 机 器 产品 和 维多利亚 时 
代 高 度 装 饰 化 风格 的 回应 。Arts & Crafts 强 调 简洁 设计 ， 而 回归 自然 是 其 整个 运动 的 核心 ， 注 重 
手工 制造 及 推崇 个 性 化 设计 ， 可 是 它 并 不 回避 使 用 现代 工具 。 这 和 我 们 现今 的 情形 有 很 多 相似 
之 处 : 世纪 之 交 ， 从 计算 机 革命 的 最 初 起 源 到 对 个 人 来 说 更 精简 、 更 意味 深长 的 事物 的 演变 ， 
以 及 对 软件 开发 技能 而 不 仅 是 生产 程序 代码 的 强调 。 

我 以 同样 的 眼光 看 待 Java; 尝试 将 程序 员 从 操作 系统 机 制 中 解放 出 来 ， 朝 着 “软件 艺 师 ”的 
方向 发 展 。 

我 和 封面 设计 者 自 孩提 时 代 就 是 朋友 ， 我 们 从 这 次 运动 中 获得 灵感 ， 并 且 都 拥有 源 自 那个 
时 期 的 〈 或 受 那个 时 期 启发 而 创作 的 ) 家 具 、 人 台灯 和 其 他 作品 。 

这 个 封面 暗示 的 另 一 主题 是 一 个 收集 盒 ， 博 物 学 家 可 以 用 它 来 展示 他 们 保存 的 昆虫 标本 。 这 
些 昆 虫 可 以 看 作 是 对 象 ， 并 放置 到 “ 盒 ” 这 个 对 象 当中 ， 而 盒 对象 又 放置 到 “封面 对 象 ”当中 ， 
这 形象 地 说 明了 面向 对 象 程序 设计 中 最 为 基本 的 “集合 ”概念 。 当 然 ， 程 序 员 可 能 会 不 禁 联想 到 
“程序 缺陷 (bug)”， 这 些 虫 子 被 捕获 ， 并 假设 在 标本 纺 中 被 杀 死 ， 最 后 禁闭 于 一 个 展示 盒 中 ， 
似乎 暗示 Java 有 能 力 发 现 、 显 示 和 制服 程序 缺陷 (事实 上 ， 这 也 是 它 最 为 强大 的 属性 之 一 )。 

在 本 版 中 ， 我 创造 了 一 幅 水 彩 画 ， 你 可 以 在 封面 的 背景 中 看 到 它 。 


致谢 


首先 感谢 和 我 一 起 开 研讨 课 、 提 供 咨询 和 开发 教学 计划 的 这 些 合作 者 : Dave Bartlett, Bill 
Venners, Chuck Allison, Jeremy Meyer 和 Jamie King。 在 我 转 而 不 停 地 竭力 为 那些 像 我 们 一 样 的 
独立 人 群 开发 在 一 起 协同 工作 的 最 佳 模式 的 时 候 ， 你 们 的 耐心 让 我 感激 不 已 。 

最 近 ， 无 疑 是 因为 有 了 Internet， 我 可 以 和 极其 众多 的 人 一 起 合作 ， 他 们 协助 我 一 起 努力 ， 
他 们 通常 是 在 家 办 公 。 过 去 ， 我 可 能 必须 为 这 些 人 提供 相当 大 的 办 公 空间 ， 不 过 由 于 现在 有 了 
网 络 、 传 真 以 及 电话 ， 我 不 需要 额外 的 开销 就 可 以 从 他 们 的 帮助 中 受益 。 在 我 尽力 学 习 更 好 地 
与 其 他 人 相处 的 过 程 中 ， 你 们 都 对 我 很 有 帮助 ， 并 且 我 希望 继续 学 习 怎 样 使 我 的 工作 能 够 通过 
借鉴 他 人 的 成 果 而 变 得 更 出 色 。Paula Steuer 在 接管 我 偶尔 的 商务 活动 时 发 挥 了 不 可 估量 的 价值 ， 
他 使 它们 变 得 井井有条 (Paula， 感 谢 你 在 我 懈怠 时 对 我 的 鞭 答 ) 。Jonathan Wilcox, Esq. 详 细 审 视 
了 我 公司 的 组 织 结构 ， 推 翻 了 每 一 块 可 能 隐藏 祸害 的 石头 ， 并 且 使 所 有 事情 都 条 理化 和 合法 化 
了 ， 这 让 我 们 心服 口服 。 感 谢 你 的 细心 和 耐心 。Sharlynn Cobaugh 使 自己 成 为 声音 处 理 的 专家 ， 
她 是 创建 多 媒体 培训 CD ROM 和 解决 其 他 问题 的 精英 成 员 之 一 。 感 谢 你 在 面临 难于 处 理 的 计算 机 
问题 时 的 坚定 不 移 。 在 布拉格 Amaio 的 人 们 也 提出 了 一 些 方案 来 帮助 我 。Daniel Will-Harris 最 先 
受到 在 网 上 工作 的 启发 ， 因 此 他 当然 是 我 所 有 设计 方案 的 主要 人 物 。 

多 年 以 来 ，Gemal Weinberg 通 过 他 的 学 术 会 议和 研讨 会 ,已 经 成 为 了 我 非 正 式 的 教练 和 导师 ， 
我 十 分 感谢 他 。 

Ervin Varga 在 第 4 版 的 技术 纠正 方面 提供 了 巨大 的 帮助 一 一 尽管 其 他 人 在 各 个 章节 和 示例 方 
面 也 帮助 良 多 ， 但 是 Ervin 是 本 书 最 主要 的 技术 复审 者 ， 他 还 承担 了 第 4 版 的 解决 方案 指南 的 重 写 
任务 。Ervin 发 现 的 错误 和 对 本 书 所 作 的 完善 对 本 书 来 说 价值 连城 。 他 对 细节 的 投入 和 关注 程度 
令 人 惊异 ， 他 是 我 所 见 过 的 远 远 超过 其 他 人 的 最 好 的 技术 读者 。 感 谢 你 ，Ervin。 

我 在 Bil Venners 的 www.Artima.com 上 的 weblog， 已 经 成 为 了 当 我 需要 交流 思想 时 的 一 种 解 
决 之 道 。 感 谢 那些 通过 提交 评论 帮助 我 证 清 概念 的 人 们 ， 包 括 James Watson, Howard Lovatt, 

















XX 


Michael Barker 以 及 其 他 一 些 人 ， 特 别 是 那些 在 泛 型 方面 提供 帮助 的 人 。 

感谢 Mark Welsh 不 懈 的 帮助 。 

Evan Cofsky 一 如 既往 地 提供 了 有 力 的 支持 ， 他 埋头 处 理 了 大 量 晓 涩 的 细节 ， 从 而 建立 和 维 
护 了 基于 Linux 的 Web 服 务 器 ， 并 保持 MindView 服 务 器 始终 处 于 协调 和 安全 的 状态 。 

一 份 特别 的 感谢 要 送 给 我 的 新 朋友 ， 咖 啡 ， 它 为 本 项 目 产生 了 几乎 无 穷 无 尽 的 热情 。 当 人 
们 来 到 MindView 研 讨 课时 ， 科 罗拉 多 州 Crested Butte 的 Camp4 Coffee 已 经 成 为 了 标准 住所 ， 并 且 
在 研讨 课 中 间 休息 期 间 ， 它 是 我 所 遇 到 的 最 好 的 饮食 场所 。 感 谢 我 的 密友 Al Smith， 是 他 使 这 里 
成 为 如 此 好 的 一 个 地 方 ， 成 为 Crested Butte 培 训 期 间 一 个 如 此 有 趣 和 愉快 的 场所 。 还 要 感谢 
Camp4 的 所 有 泡 吧 常客 们 ， 很 高 兴 他 们 总 是 为 我 们 提供 一 些 饮料 。 

感谢 Prentice Hall 的 人 们 不 断 地 为 我 提供 我 所 需要 的 一 切 ， 并 容忍 我 所 有 的 特殊 需求 ， 而 且 
不 厌 其 烦 地 帮 我 把 所 有 事情 都 搞定 。 

在 我 的 开发 过 程 中 ， 有 些 工具 已 经 被 证 明 是 无 价 的 ， 但 每 次 使 用 它们 时 都 会 非常 感激 它们 
的 创建 者 。Cygwin (http://www.cygwin.com)) 为 我 解决 了 无 数 Windows 不 能 解决 的 问题 ， 并 且 每 天 
我 都 会 变 得 更 加 依赖 它 (如 果 在 15 年 前 当 我 的 头脑 因 使 用 Gnu Emacs 而 搞 得 发 懂 的 时 候 ， 能 有 这 
些 该 多 好 啊 )。IBM 的 Eclipse (http://www.eclipse.org) 对 开发 社区 做 出 了 真正 杰出 的 贡献 ， 并 且 随 
着 它 的 不 断 升级 ， 我 期 望 能 看 到 它 的 更 伟大 之 处 (IBM 是 怎样 成 为 潮流 所 向 的 ? 我 肯定 错过 了 一 
份 备忘录 )。 而 JetBrains IntelliJ Idea 则 继续 开阔 着 开发 工具 的 创新 之 路 。 

我 一 开始 就 将 Sparxsystems 的 Enterprise Architecture 用 于 本 书 ， 并 且 它 很 快 就 成 为 了 我 选择 的 
UML 工 具 。Marco Hunsicker 的 Jalopy 代 码 格式 化 器 (www.triemax.com) 在 大 量 的 场合 都 派 上 了 
用 场 ， 而 且 Marco 在 将 其 配置 成 满足 我 的 特殊 需求 方面 也 提供 了 大 量 的 帮助 。 我 还 发 现 Slava 
Pestov 的 JEdit 及 其 插件 经 常会 显得 很 有 用 (wwwjeditorg)， 并 且 对 于 研讨 课 来 说 ， 它 是 非常 适合 
初学 者 的 编辑 器 。 

当然 ， 如 果 我 在 其 他 地 方 强调 得 还 不 够 的 话 ， 我 得 再 次 重申 ， 我 经 常 使 用 Python 
(www.Python.org) 解决 问题 ， 在 我 的 密友 Guido Van Rossum 和 PythonLabs 那 些 身材 腾 肿 恩 笨 的 天 
才 人 物 的 智慧 结晶 的 基础 上 ， 我 花费 了 好 几 天 的 时 间 进 行 冲刺 (Tim Peters， 我 现在 已 经 把 你 借 
的 鼠标 加 了 个 框 ， 正 式 命名 为 TimBotMouse)。 你 们 这 伙 人 必须 到 更 健康 的 地 方 去 吃 午餐 。( 还 要 
感谢 整个 Python 社区 ， 他 们 是 一 帮 令 人 吃惊 的 群体 。) 

很 多 人 向 我 发 送 修正 意见 ， 我 感激 所 有 这 些 人 ， 第 1 版 特别 要 感谢 ，Kevin Raulerson (RA 
无 数 的 程序 缺陷 )，Bob Resendes (简直 难以 置信 ) John Pinto, Joe Dante, Joe Sharp (三 位 都 难 
以 置信 ) David Combs (校正 了 许多 语法 和 声明 ) Dr. Robert Stephenson, John Cook, Franklin 
Chen, Zev Griner, David Karr, Leander A. Stroschein, Steve Clark, Charles A, Lee, Austin 
Maher, Dennis P. Roth, Roque Oliveira, Douglas Dunn, Dejan Ristic, Neil Galameau, David B. 
Malkovsky, Steve Wilkinson 以 及 许 许 多 多 的 人 。 本 书 第 1 版 在 欧洲 发 行 时 ，Marc Meurrens 在 电子 
版 宣传 和 制作 方面 做 出 了 巨大 的 努力 。 

感谢 在 本 书 第 2 版 中 使 用 Swing 类 库 帮助 我 重新 编写 示例 的 人 们 , 以 及 其 他 助手 一 Jon Shvarts、 
Thomas Kirsch、Rahim Adatia、Rajesh Jain、Ravi Manthena、Banu Rajamani、Jens Brandt、Nitin 
Shivaram、Malcolm Davis， 还 有 所 有 表示 支持 的 人 。 

在 第 4 版 中 ，Chris Grindstaff 对 SWT 一 节 的 撰写 提供 很 多 帮助 ， 而 Sean Neville 为 我 撰写 了 Flex 
一 节 的 第 一 稿 。 
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每 当 我 认为 我 已 经 理解 了 并 发 编程 时 ， 又 会 有 新 的 奇 山 险峰 等 待 我 去 征服 。 感 谢 Brian Goetz 
帮助 我 克服 了 在 撰写 新 版 本 的 “并 发 ”一 章 时 遇 到 的 种 种 艰难 险阻 ， 并 发 现 了 其 中 所 有 的 缺陷 
(我 希望 如 此 ! ) 

对 Delphi 的 理解 使 我 更 容易 理解 Java， 这 一 点 儿 都 不 奇怪 ， 因 为 它们 有 许多 概念 和 语言 设计 
决策 是 相通 的 。 我 的 懂 Delphi 的 朋友 们 给 我 提供 了 许多 帮助 ， 使 我 能 够 洞察 一 些 非凡 的 编程 环境 。 
他 们 是 Marco Cantu ( 另 一 个 意大利 人 -难道 会 说 拉丁 语 的 人 在 学 习 Java 时 有 得 天 独 厚 的 优势 ? )、 
Neil Rubenking (直到 发 现 喜欢 计算 机 之 前 ， 他 一 直 都 在 做 瑜珈 /素食 / 祥 道 )， 当 然 还 有 Zack 
Urlocker (最 初 的 Delphi 产 品 经 理 )， 他 是 我 游历 世界 时 的 好 伙伴 。 我 们 都 很 感激 Anders Hejlsberg 
的 卓越 才华 ， 他 在 C# 领 域 不 懈 地 奋斗 着 《正如 你 将 在 本 书 中 看 到 的 ，C# 是 Java SE5 主 要 的 灵感 
z=) 

我 的 朋友 Richard Hale Shaw (以 及 Kim) 的 洞察 力 和 支持 都 很 有 帮助 。Richard 和 我 花 了 数 月 
时 间 将 教学 内 容 合并 到 一 起 ， 并 为 参加 学 习 的 学 生 设 计 出 一 套 完美 的 学 习 体验 。 

书籍 设计 、 封 面 设计 以 及 封面 照片 是 由 我 的 朋友 Daniel Wil-Harris 制 作 的 。 他 是 一 位 著名 的 
作家 和 设计 家 (http://www.WillHamis.com) ， 在 计算 机 和 桌面 排版 发 明之 前 ， 他 在 初中 的 时 候 就 
常常 摆弄 刊 擦 信 (rub-on letter) ， 他 总 是 抱怨 我 的 代数 含糊 不 清 。 然 而 ， 要 声明 的 是 ， 是 我 自己 
制作 的 照排 好 的 (camera-ready) 页 面 ， 所 以 所 有 排 字 错误 都 应 该 算 到 我 这 里 。 我 是 用 Microsoft 
Word XP for Windows 来 编写 这 本 书 的 ， 并 使 用 Adobe Acrobat 制 作 照 排 页 面 的 。 本 书 是 直接 从 
Acrobat PDF 文件 创建 而 来 的 。 电 子 时 代 给 我 们 带 来 了 厚礼 ， 我 恰巧 是 在 海外 创作 了 本 书 第 1 版 和 
第 ?版 的 最 终 稿 一 第 1 版 是 在 南非 的 开 普 敦 送出 的 ， 而 第 2 版 却 是 在 布拉格 寄 出 的 。 第 3 版 和 第 4 版 
则 来 自 科 罗拉 多 州 的 Crested Butte。 正 文字 体 是 Georgia， 而 标题 是 Verdana。 封 面 字体 是 ITC 
Rennie Machintosh, 

“特别 感谢 我 的 所 有 老师 和 我 的 所 有 学 生 (他 们 也 是 我 的 老师 )。 

Molly， 在 我 从 事 这 一 版 的 写作 时 总 是 坐 在 我 腿 上 ， 为 我 提供 了 她 特有 的 温 软 而 毛 萌 萌 的 支持 。 

曾 向 我 提供 过 支持 的 朋友 包括 (当然 还 不 止 他 们 ): Patty Gast(Masseuse extraordinary), 
Andrew Binstock, SteveSinofsky, JD Hildebrandt, Tom Keffer, Brian McElhinney, Brinkley Barr, 
(Midnight Engineering》 杂 志 社 的 Bil Gates, Larry Constantine 和 Lucy Lockwood, Gene Wang, 

”Dave Mayer, David Intersimone，Chris 和 Laura Strand, Almquists, Brad Jerbic, Marilyn Cvitanic, 
Mark Mabry, Dave Stoner, Cranstons, Larry Fogg, Mike Sequeira, Gary Entsminger, Kevin 和 
Sonda Donovan, Joe Lordi, Dave 和 Brenda Bartlett, Patti Gast, Blake, Annette&Jade, 
Rentschlers, Sudeks, Dick, Patty#flLee Eckel，Lynn 和 Todd 以 及 他 们 的 家 人 。 当 然 还 有 我 的 父亲 
和 母亲 。 








绪 论 


“上 帝 赋予 人 类 说 话 的 能 力 ， 而 言语 又 创造 了 思想 ， 思 想 是 人 类 对 宇宙 的 量度 。” 
一 一 摘自 《Prometheus Unbound), Shelley 


人 类 …… 极 其 受 那些 已 经 成 为 社会 表达 工具 的 特定 语言 的 支配 。 想 像 一 下 ， 如 果 一 个 人 可 

以 不 使 用 语言 就 能 够 从 本 质 上 适应 现实 世界 ， 语 言 仅 仅 是 解决 具体 的 交流 和 反映 问题 时 偶尔 才 

用 到 的 方式 ， 我 们 会 发 现 ， 这 只 能 是 一 种 幻想 。 事 实 上 ,“ 真 实 世界 ”在 很 大 程度 上 是 不 知 不 觉 
地 基于 群体 的 语言 习惯 形成 的 。 

一 一 摘自 《The Status of Linguistics As A Science), 1929, Edward Sapir 


如 同 任何 人 类 语言 一 样 ，Java 提 供 了 一 种 表达 概念 的 方式 。 如 果 使 用 得 当 ， 随 着 问题 变 得 更 
庞大 更 复杂 ， 这 种 表达 工具 将 会 比 别 的 可 供 选择 的 语言 更 为 简单 、 灵 活 。 

我 们 不 应 该 将 Java 仅 仅 看 作 是 一 些 特性 的 集合 一 -有 一 些 特性 在 孤立 状态 下 没有 任何 意义 。 
只 有 在 考虑 到 设计 ， 而 不 仅仅 是 编码 时 ， 才 能 完整 地 运用 Java 的 各 部 分 。 而 且 ， 要 按照 这 种 方式 
来 理解 Java， 必 须 理解 在 语言 和 编程 中 经 常 碰 到 的 问题 。 这 本 书 讨论 的 是 编程 问题 ， 它 们 为 什么 
成 为 问题 ， 以 及 Java 已 经 采取 什么 样 的 方案 来 解决 它们 。 因 此 ， 每 章 所 闹 述 的 特性 集 ， 都 是 基于 
我 所 看 到 的 这 一 语言 在 解决 特定 类 型 问题 时 的 方式 。 按 照 这 种 方式 ， 我 希望 能 够 每 次 引导 读者 
前 进 一 点 ， 直 到 Java 思 想 意识 成 为 你 最 自然 不 过 的 语言 。 

自始至终 ， 我 一 直 持 这 样 的 观点 : 你 需要 在 头脑 中 创建 一 个 模型 ， 以 加 强 对 这 种 语言 的 深 
入 理解 ， 如 果 你 遇 到 了 疑问 ， 就 将 它 反馈 到 头脑 中 的 模型 并 推断 出 答案 。 


前 提 条 件 


本 书 假定 你 对 程序 设计 有 一 定 程度 的 熟悉 ， 你 已 经 知道 程序 是 一 些 语句 的 集合 ， 知 道子 程 
序 /函数 / 宏 的 概念 ， 知 道 像 “if” 这 样 的 控制 语句 和 像 “while” 这 样 的 循环 结构 ， 等 等 。 不 过 ， 
你 可 能 在 许多 地 方 已 经 学 到 过 这 些 ， 例 如 使 用 宏 语言 进行 程序 设计 ， 或 者 使 用 像 Perl 这 样 的 工具 
工作 。 只 要 你 已 经 达到 能 够 自如 地 运用 程序 设计 基本 思想 的 程度 ， 你 就 能 够 顺利 阅读 本 书 。 当 
然 ， 本 书 对 C 程 序 员 来 说 更 容易 ， 对 于 C++ 程序 员 更 是 如 此 ， 但 是 ， 即 使 你 没有 实践 过 这 两 种 语 
言 ， 也 不 要 否定 自己 一 而 应 该 更 加 努力 学 习 。 并 且 ， 从 wwwMindView .net 处 可 下 载 的 《Thinking 
in C》 多 媒体 研讨 课 能 够 带领 你 快速 学 习 所 必需 的 Java 基 础 知识 ) 。 不 过 ， 我 还 会 介绍 面向 对 象 
(OOP) 的 概念 和 Java 的 基本 控制 机 制 。 

尽管 本 书 可 能 会 经 常 引用 、 参 考 C 和 C++ 语言 的 特性 ， 但 这 并 不 是 打算 让 它们 成 为 内 部 注释 ， 
而 是 要 帮助 所 有 的 程序 员 正确 看 待 这 些 语言 ， 毕 竞 Java 是 从 这 些 语言 衍生 而 来 的 。 我 会 努力 简化 
这 些 引用 、 参 考 ， 并 且 对 那些 我 认为 一 个 非 CIC++ 程 序 员 可 能 不 太 熟 悉 的 地 方 加 以 解释 。 


学 习 Java 





大 概 在 我 的 第 一 本 书 《Using C++》(Osbome/McGraw-Hill，1989) 出 版 发 行 的 同一 时 候 ， 我 
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就 开始 教授 这 种 语言 了 。 讲 授 程 序 设计 语言 已 经 成 为 我 的 职业 了 ， 自 1987 年 以 来 ， 我 在 世界 各 
地 的 听众 中 看 到 ， 有 的 型 昏 欲 睡 ， 有 的 面 无 表情 ， 有 的 表情 迷茫 。 当 我 开始 给 一 些小 团体 进行 
室内 培训 时 ， 在 这 些 实践 中 我 发 现 了 一 些 事情 。 即 使 那些 面 带 微笑 频频 点 头 的 人 也 对 很 多 问题 
心 存 困惑 。 我 发 现 ， 多 年 来 在 软件 开发 会 议 上 由 我 主持 的 C++ 分 组 讨论 会 (后 来 变 成 Java 分 组 讨 
论 会 ) 中 ， 我 和 其 他 的 演讲 者 往往 是 在 极 短 的 时 间 内 告诉 听众 许多 话题 。 因 此 ， 最 后 由 于 听众 
的 水 平 不 同和 讲授 教材 的 方式 这 两 方面 的 原因 ， 我 可 能 最 终 会 失去 一 部 分 听众 。 可 能 这 样 要 求 
得 太 多 了 ， 但 因为 我 是 传统 演讲 的 反对 者 之 一 (而 且 对 于 大 多 数 人 来 说 ， 我 相信 这 种 抵制 是 因 
为 厌倦 )， 因 此 我 想 尽力 让 每 个 人 都 可 以 跟 得 上 演讲 的 进度 。 

我 曾经 一 度 在 相当 短 的 时 间 内 做 了 一 系列 不 同 的 演讲 。 因 此 ， 我 结束 了 “实践 和 和 迭代 ”( 一 
项 在 Java 程 序 设计 中 也 得 到 很 好 运用 的 技术 ) 的 学 习 。 最 后 ， 我 根据 自己 在 教学 实践 中 学 到 的 东 
西 发 展 出 一 门 课程 。 我 的 公司 一 MindView 有 限 公司 现在 提供 公开 的 室内 “Thinking in Java” 研 讨 
课 ， 这 是 我 们 主要 的 初级 研讨 课 ， 为 以 后 更 高 级 的 研讨 课 提供 基础 。 读 者 可 以 到 网 站 
www.MindView.net 上 了 解 详细 情况 。( 初 级 研讨 课 在 “Hands-On Java” 光 盘 上 也 能 找到 。 上 述 网 
站 也 可 以 找到 相关 信息 。) 

从 每 个 研讨 课 获得 的 反馈 信息 都 可 以 帮助 我 去 修改 和 重新 制定 课程 教材 ， 直 到 我 认为 它 能 
够 成 为 一 个 良性 运转 的 教学 工具 为 止 。 不 过 不 能 将 本 书 视 为 一 般 的 研讨 课 笔 记 ， 我 努力 在 本 书 
中 放 入 尽 可 能 多 的 信息 ， 并 且 合 理 地 组 织 本 书 结构 ， 从 而 引导 读者 顺利 进入 下 一 主题 。 最 重要 
的 是 ， 本 书面 向 那些 孤军 奋战 一 门 新 的 程序 设计 语言 的 读者 。 


目标 


就 像 我 前 一 本 书 《Thinking in C++》 那 样 ， 在 设计 本 书 时 ， 我 脑子 里 始终 思考 的 一 件 事情 就 
是 : 人 们 学 习 语言 的 方式 。 当 我 思索 书 中 的 一 章 时 ， 我 思索 的 是 如 何在 研讨 课 上 教 好 一 堂 课 。 
研讨 课 听 众 的 反馈 意见 帮助 我 理解 了 哪些 是 需要 详细 了 明 的 有 难度 的 部 分 。 在 某 些 领域 ， 我 一 
开始 雄心 勃勃 ， 在 其 中 一 下 子囊 括 了 过 多 的 特性 ， 后 来 通过 在 讲解 这 些 材料 的 过 程 中 ， 我 逐渐 
意识 到 如 果 要 囊括 过 多 的 特性 ， 就 必须 对 它们 全 部 解释 清楚 ， 而 这 很 容易 使 学 生产 生 混淆 。 

因此 ， 本 书 的 每 一 章 都 设法 只 教授 一 个 特性 ， 或 者 一 小 组 互相 关联 的 特性 ， 并 且 不 会 依赖 
于 还 未 介绍 的 概念 。 通 过 这 种 方式 ， 你 可 以 在 你 当前 所 掌 提 的 知识 背景 下 ， 在 继续 向 前 学 习 之 
前 ， 消 化 吸收 每 一 部 分 内 容 。 

在 这 本 书 中 我 想 达 到 的 目标 是 : 

1) 每 一 次 只 演示 一 个 步骤 的 材料 ， 以 便 读者 在 继续 后 面 的 学 习 之 前 可 以 很 容易 地 消化 吸收 
每 一 个 观念 。 仔 细 地 对 特性 的 讲解 进行 排序 ， 以 使 得 你 在 看 到 对 某 个 特性 的 运用 之 前 ， 会 先 了 
解 它 。 当 然 ， 这 并 非 总 是 可 行 的 ， 在 那些 不 可 行 的 情况 下 ， 会 给 出 一 个 简短 的 介绍 性 描述 。 

2) 使 用 的 示例 尽 可 能 简单 、 短 小 。 这 样 做 有 时 会 妨碍 我 们 解决 “真实 世界 ”的 问题 ， 但 是 ， 
我 发 现 对 于 初学 者 ， 能 够 理解 例子 的 每 一 个 细节 ， 而 不 是 理解 它 所 能 够 解决 的 问题 范畴 ， 前 者 
通常 更 能 为 他 们 带 来 愉悦 。 同 样 ， 适 合 在 教室 内 学 习 的 代码 数量 也 有 严格 限制 。 正 因为 如 此 ， 
我 将 毫 无 疑问 地 会 遭 到 批评 -批评 我 使 用 “玩具 般 的 示例 "， 但 是 我 乐意 接受 那些 有 利于 为 教育 带 
来 益处 的 种 种 事物 。 

3) 向 读者 提供 “我 认为 对 理解 这 种 程序 设计 语言 来 说 很 重要 ”的 部 分 ， 而 不 是 提供 我 所 知 
道 的 所 有 事情 。 我 相信 信息 在 重要 性 上 存在 层次 差别 ， 有 一 些 事实 对 于 95% 的 程序 员 来 说 永远 
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不 必 知道 一 那些 只 会 困扰 他 们 并 且 使 他 们 对 程序 复杂 性 平添 许多 感触 。 举 一 个 C 语 言 的 例子 ， 如 
果 能 够 记 住 操作 符 优先 表 (我 从 未 记 住 过 )， 那 么 可 以 写 出 灵巧 的 代码 。 但 是 你 要 再 想 一 想 ， 这 
样 做 会 给 读者 /维护 者 带 来 困惑 。 因 此 忘掉 优先 权 ， 在 不 是 很 清楚 的 时 候 使 用 圆 括号 就 行 了 。 

4) 使 每 部 分 的 重点 足够 明确 ， 以 便 缩短 教学 和 练习 之 间 的 时 间 。 这 样 做 不 仅 使 听众 在 亲身 
参与 研讨 课时 思维 更 为 活跃 和 集中 ， 而 且 还 可 以 让 读者 更 具有 成 就 感 。 

5) 给 读者 打下 坚实 的 基础 ， 使 读者 能 够 充分 理解 问题 ， 以 便 转 入 更 难 的 课程 学 习 和 书籍 阅 
读 中 。 
根据 本 书 教学 


本 书 最 初 的 版 本 是 从 一 个 为 期 一 周 的 研讨 课 演变 而 来 的 ， 当 时 Java 还 处 于 初级 阶段 ， 因 此 
一 周 已 经 足以 覆盖 Java 的 语言 特性 了 。 随 着 Java 的 成 长 ， 有 越 来 越 多 的 特性 和 类 库 不 断 地 添加 了 
进来 ， 我 固执 地 试图 仍旧 在 一 周 内 教授 所 有 的 内 容 。 那 时 ， 有 一 位 顾客 请 我 讲课 ， 内 容 “ 只 包 
括 基础 知识 ”"， 我 教授 的 过 程 中 ， 我 发 现在 一 周 的 时 间 内 填 鸭 式 的 教授 所 有 的 内 容 ， 对 于 我 自己 
和 参加 研讨 课 的 人 来 说 ， 都 是 一 种 痛苦 。Java 已 经 不 再 是 一 种 可 以 在 一 周 内 教授 的 “简单 ” 语 
言 了 。 

这 份 精力 和 感悟 在 极 大 程度 上 促使 我 对 本 书 进行 了 重新 的 组 织 ， 现 在 它 已 经 被 设计 为 可 以 
支撑 一 个 两 周 的 研讨 课 ， 或 者 是 一 门 两 学 期 的 大 学 课程 。 介 绍 性 的 部 分 在 “通过 异常 处 理 错误 ” 
一 章 就 结束 了 ， 但 是 你 可 能 还 想 补 充 了 解 一 些 对 JDBC、Servlet 和 JSP 的 介绍 ， 这 些 内 容 构成 了 另 
外 一 门 基础 课程 ， 即 Hands-on Java 光盘 的 核心 内 容 。 本 书 剩余 部 分 可 以 组 成 一 门 中 级 课程 ， 即 
Intermediate Thinking in Java 光 盘 中 所 包含 的 材料 。 这 两 张 光盘 在 www.MindView.net 都 有 售 。 

通过 www.prenhallprofessional.com 与 Prentice-Hall 联 系 ， 可 以 得 到 能 够 教授 本 书 这 些 材料 的 教 
师 信息 。 


JDK 的 HTML 文 档 


Sun 公 司 的 Java 语 言及 其 类 库 (可 以 从 java.sun.com 免 费 下 载 ) 配套 提供 了 电子 版 文档 , 可 使 
用 Web 浏 览 器 阅读 。 许 多 出 版 的 Java 书 籍 中 也 都 有 这 份 文档 的 备份 。 你 可 能 已 经 拥有 了 它 ， 或 者 
能 够 下 载 ， 所 以 除非 必要 ， 本 书 不 会 再 重复 那 份 文档 。 因 为 一 般 来 说 ， 用 Web 浏 览 器 查找 类 的 描 
述 比 在 书 中 查找 要 快 得 多 (并且 在 线 文 档 更 能 保持 更 新 )。 你 仅 需要 参考 “JDK 文 档 "。 只 有 当 需 
要 对 文档 进行 补充 ， 以 便 你 能 够 理解 特定 实例 时 ， 本 书 才 会 提供 有 关 类 的 一 些 附加 说 明 。 


练习 


在 研讨 课 上 ， 我 发 现 一 些 简单 的 练习 非常 利于 学 生 们 理解 掌握 有 关 概 念 ， 因 此 在 每 一 章 的 
最 后 都 安排 了 一 些 习题 。 

大 多 数 练习 设计 得 都 很 简单 ， 可 以 让 学 生 在 课堂 上 在 合理 的 时 间 内 完成 这 些 作业 ， 以 便 指 
导 老师 检查 辅导 以 确保 所 有 的 学 生 都 吸收 了 教材 的 内 容 。 有 一 些 题目 具有 挑战 性 ， 但 并 没有 难 
度 很 高 的 题目 。 : 

一 些 经 过 挑选 的 练习 答案 可 以 在 名 为 The Thinking in Java Annotated Solution Guide 的 电子 文 
档 中 找到 ， 仅 需 少许 费用 便 可 以 从 www.MindView.net 下 载 得 到 。 





Java 基 础 


本 书 还 附送 可 从 www.MindView.net 处 下 载 的 免费 的 多 媒体 研讨 课 。 这 是 《Thinking in C) 的 
研讨 课 ， 它 介绍 了 Java 语 法 沿用 的 C 语 言 中 的 语法 、 操 作 符 及 函数 。 在 本 书 以 前 的 版 本 中 ， 这 部 
分 内 容 收 录 在 随 书 附送 的 “Java 基 础 ”CD 中 ， 但 是 现在 这 个 研讨 课 可 以 免费 下 载 了 。 

我 原本 打算 让 Chuck Allison 把 “Thinking in C” 创 建成 一 个 独立 产品 ， 不 过 我 还 是 决定 将 它 
和 第 2 版 的 《Thinking in C++》， 以 及 第 2 版 和 第 3 版 的 《Thinking in Java》 包 含 在 一 起 ， 这 样 做 是 
为 了 让 参加 研讨 课 的 、 没 有 太 多 C 语 言 基 本 语法 背景 的 人 们 能 够 很 方便 地 找到 相关 资料 。 应 该 抛 
开 这 种 思想 ;“ 我 是 一 个 聪明 的 程序 员 ， 我 不 想 学 习 C， 而 想 学 习 C++ 或 lava， 因 此 我 会 跳 过 C 直 
接 到 C++/Java。” 在 到 了 研讨 课 上 后 ， 这 些 人 渐渐 明白 ， 很 好 地 理解 C 语 言语 法 这 个 先决 条 件 很 

技术 在 不 断 发 生变 化 ， 将 《Thinking in C》 重 新 制作 为 可 下 载 的 Flash 形 式 比 将 其 收录 在 CD 
中 要 更 具 实际 意义 。 通 过 在 线 提供 这 个 研讨 课 ， 我 可 以 保证 每 个 人 都 可 以 事先 做 好 充足 的 准备 。 

(Thinking in C) 研讨 课 也 让 本 书 获 得 了 更 多 的 读者 。 尽 管 本 书 中 “操作 符 ” 和 “控制 执行 
流程 ”两 章 履 盖 了 Java 继 承 自 C 的 基本 部 分 ， 但 是 在 线 研讨 课 仍旧 是 更 好 的 介绍 ， 而 且 它 要 求学 
生 所 具备 的 程序 设计 背景 比 这 本 书 要 求 的 还 要 少 。 


源 代码 


本 书 的 所 有 源 代码 都 能 以 保留 版 权 的 免费 软件 的 形式 得 到 ， 它 们 是 以 单一 包 的 形式 发 布 的 ， 
访问 www.MindView.net 网 站 便 可 获取 。 为 了 确保 你 获得 的 是 最 新 版 本 ， 这 个 发 布 这 些 源 代码 和 
本 书 电子 版 的 网 站 是 一 个 官方 网 站 。 你 可 以 在 课堂 或 其 他 教育 场所 发 布 这 些 代码 。 

保留 版 权 的 主要 目的 是 为 了 确保 源 代码 能 够 被 正确 地 引用 ， 并 且 防 止 在 未 经 许可 的 情况 下 ， 
在 出 版 媒体 中 重新 发 布 这 些 代码 (只 要 说 明 是 引用 了 这 些 代 码 ， 那 么 在 大 多 数 媒 介 中 使 用 本 书 
中 的 示例 通常 不 是 问题 )。 

在 每 个 源码 文件 中 ， 都 包含 下 述 版 权 声 明文 字 : 

1/:! Copyright. txt 


This computer source code is Copyright ©2006 MindView, Inc. 
ALL Rights Reserved. 


Permission to use, copy. modify, and distribute this 
computer source code (Source Code) and its documentation 
without fee and without a written agreement for the 
purposes set forth below is hereby granted, provided that 
the above copyright notice, this paragraph and the 
following five numbered paragraphs appear in all copies. 


1. Permission is granted to compile the Source Code and to 
include the compiled code, in executable format only, in 
Personal and commercial software programs. 


2. Permission is granted to use the Source Code without 
modification in classroom situations, including in 
presentation materials, provided that the book “Thinking in 
Java" is cited as the origin. 


3. Permission to incorporate the Source Code into printed 
media may be obtained by contacting: 


MindView, Inc. 5343 Valle Vista La Mesa, California 91941 
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Wayne@MindView.net 


4. The Source Code and documentation are copyrighted by 
MindView, Inc. The Source code is provided without express 
or implied warranty of any kind, including any implied 
warranty of merchantability, fitness for a particular 
Purpose or non-infringement. MindView, Inc. does not 
warrant that the operation of any program that includes the 
Source Code will be uninterrupted or error-free. MindView, 
Inc. makes no representation about the suitability of the 
Source Code or of any software that includes the Source 
Code for any purpose. The entire risk as to the quality 
and performance of any program that includes the Source 
Code is with the user of the Source Code. The user 
understands that the Source Code was developed for research 
and instructional purposes and is advised not to rely 
exclusively for any reason on the Source Code or any 
program that includes the Source Code. Should the Source 
Code or any resulting software prove defective, the user 
assumes the cost of all necessary servicing, repair, or 
correction. 


5. IN NO EVENT SHALL MINDVIEW, INC., OR ITS PUBLISHER BE 
LIABLE TO ANY PARTY UNDER ANY LEGAL THEORY FOR DIRECT, 
INDIRECT, SPECIAL, INCIDENTAL, OR CONSEQUENTIAL DAMAGES, 
INCLUDING LOST PROFITS, BUSINESS INTERRUPTION, LOSS OF 
BUSINESS INFORMATION, OR ANY OTHER PECUNIARY LOSS, OR FOR 
PERSONAL INJURIES, ARISING OUT OF THE USE OF THIS SOURCE 
CODE AND ITS DOCUMENTATION, OR ARISING OUT OF THE INABILITY 
TO USE ANY RESULTING PROGRAM, EVEN IF MINDVIEW, INC., OR 
ITS PUBLISHER HAS BEEN ADVISED OF THE POSSIBILITY OF SUCH 
DAMAGE. MINDVIEW, INC. SPECIFICALLY DISCLAIMS ANY 
WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED 
WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR 
PURPOSE. THE SOURCE CODE AND DOCUMENTATION PROVIDED 
HEREUNDER IS ON AN “AS IS" BASIS, WITHOUT ANY ACCOMPANYING 
SERVICES FROM MINDVIEW, INC., AND MINDVIEW. INC. HAS NO 
OBLIGATIONS TO PROVIDE MAINTENANCE, SUPPORT, UPDATES, 
ENHANCEMENTS, OR MODIFICATIONS. 


Please note that MindView, Inc. maintains a Web site which 
is the sole distribution point for electronic copies of the 
Source Code, http://www.MindView.net (and official mirror 
sites), where it is freely available under the terms stated 
above. 


If you think you've found an error in the Source Code, 
please submit a correction using the feedback system that 
you will find at http://www. MindView.net. 

MES 


你 可 以 在 自己 的 项 目 中 引用 这 些 代码 ， 也 可 以 在 课堂 上 引用 它们 (包括 你 的 演示 材料 )， 只 
要 保留 每 个 源 文件 中 出 现 的 保留 版 权 声明 即 可 。 


编码 标准 
在 本 书 的 正文 中 ， 标 识 符 方法、 变量 和 类 名 ) 排 为 粗 体 。 大 多 数 关键 字 也 排 为 粗 体 ， 但 
是 不 包括 那些 频繁 使 用 的 关键 字 ， 例 如 “class"， 因 为 如 果 将 它们 也 设 为 粗 体会 令 人 十 分 厌烦 。 
对 于 本 书 中 的 示例 ， 我 使 用 了 一 种 特定 的 编码 格式 ， 此 格式 尽 可 能 地 遵循 了 Sun 自 己 在 所 有 
代码 中 实际 使 用 的 格式 ， 在 它 的 网 站 上 你 会 发 现 这 些 代码 (ML java.sun.com/docs/codeconv/ 


index.html) ， 并 且 似乎 大 多 数 Java 开 发 环境 都 支持 这 种 格式 。 如 果 你 已 经 读 过 我 的 其 他 著作 ， 你 
会 注意 到 Sun 的 编码 格式 与 我 的 一 致 -尽管 这 与 我 没什么 关系 (我 了 解 这 一 点 ) ， 但 我 还 是 很 高 兴 。 
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对 代码 进行 格式 化 这 个 议题 常常 会 招致 几 个 小 时 的 热烈 争论 ， 因 此 我 不 会 试图 通过 自己 的 示例 


来 规定 正确 的 格式 ， 我 对 自己 使 用 的 格式 有 自己 的 想法 。 因 为 Java 是 一 种 自由 形式 的 程序 设计 语 


言 ， 所 以 你 可 以 继续 使 用 自己 喜欢 的 格式 。 编 码 风格 问题 的 一 种 解决 方案 是 使 用 像 
Jalopy(www.triemax.com) 这 样 的 工具 来 将 格式 转变 为 适合 你 的 形式 ， 该 工具 帮助 我 撰写 了 此 书 。 
本 书 中 打印 的 代码 文件 都 用 一 个 自动 系统 进行 过 测试 ， 应 该 全 部 都 能 够 运行 ， 而 且 无 编译 
错误 。 
本 书 聚焦 于 Java SE5/6， 并 用 它们 进行 过 测试 。 如 果 你 需要 学 习 本 书 这 一 版 中 没有 讨论 的 
Java 语 言 的 先前 版 本 ， 可 以 从 www.MindView.net 处 免费 下 载 本 书 的 第 1 版 到 第 3 版 。 


错误 
无 论 作者 使 用 多 少 技巧 去 查找 错误 ， 但 是 有 些 错误 还 是 悄悄 地 潜藏 了 起 来 ， 并 且 经 常 对 新 


读者 造成 困扰 。 如 果 你 发 现 了 任何 你 确信 是 错误 的 东西 ， 请 使 用 在 www.MindView.net 处 可 以 找 
到 的 为 本 书 专 设 的 链接 来 提交 错误 以 及 你 建议 的 修正 。 对 你 的 帮助 我 将 不 胜 感激 。 
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第 1 章 对 象 导论 
1.1 抽象 过 程 
1.2 每 个 对 象 都 有 一 个 接口 
1.3 每 个 对 象 都 提供 服务 … 
1.4 被 隐藏 的 具体 实现 
1.5 复 用 具体 实现 …… 
1.6 继承 oe 

1.6.1 “是 一 个 ”与 “ 像 是 一 个 ”关系 
17 伴随 多 态 的 可 互 换 对 象 … 
1.8 单 根 继承 结构 
19 容器 …… 

1.9.1 参数 化 类 型 
1.10 对 象 的 创建 和 生命 期 
1.11 异常 处 理 : 处 理 错误 
112 并 发 编程 … 
1.13 Java 与 Internet 

1.13.1 Web 是 什么 

1.13.2 客户 端 编程 … 

1.13.3 服务 器 端 编程 
1.14 总 结 

第 2 章 一 切 都 是 对 象 
21 用 引用 操纵 对 象 
2.2 必须 由 你 创建 所 有 对 象 
2.2.1 存储 到 什么 地 方 
2.2:2 特例 : 基本 类 型 
2.2.3 Java 中 的 数组 … 
2.3 永远 不 需要 销毁 对 象 
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2.3.1 作用 域 ……… 
2.3.2 对 象 的 作用 域 
2.4 创建 新 的 数据 类 型 : 类 
2.4.1 字段 和 方法 e 
25 方法 、 参 数 和 返回 值 
2.5.1 参数 列表 …… 
2.6 构建 一 个 Java 程 序 
2.6.1 名 字 可 见 性 … 
2.6.2 运用 其 他 构件 
2.6.3 static 关键 字 … 
2.7 你 的 第 一 个 Java 程 序 
2.7.1 编译 和 运行 … 
2.8 注释 和 伐 入 式 文档 
2.8.1 注释 文档 
2.8.2 语法 …… 
2.8.3 代入 式 HTML 
2.8.4 一 些 标签 示例 






















2.8.5 文档 示例 … 5 
2.9 编码 风格 6 
2.10 总 结 
2.11 练习 ` 

第 3 章 操作 符 8 


3.1 更 简单 的 打印 语句 、 
3.2 使 用 Java 操 作 符 
3.3 优先 级 

3.4, 赋值 … 
3.4.1 方法 调用 中 的 别名 问题 ， 
3.5 BREH 
3.5.1 一 元 加 、 减 操作 符 
3.6 自动 递增 和 递减 “ 
3.7 关系 操作 符 …… 
3.7.1 测试 对 象 的 等 价 性 












3.9.1 指数 记 数 法 … 





3.10 按 位 操作 符 
3.11 移 位 操作 符 
3.12 三 元 操作 符 if-else 
3.13 字符 串 操作 符 + 和 += 
3.14 使 用 操作 符 时 常 犯 的 错误 > 
3.15 类 型 转换 操作 符 

3.15.1 截 尾 和 伟人 
3.15.2 提升 ……… 
3.16 Java 没 有 sizeof 
3.17 操作 符 小 结 
3.18 总 结 ee 

第 4 章 控制 执行 流程 
4.1 true 和 false 
4.2 if-else 
4.3 eH 

4.3.1 do-while 








































4.3.3 逗号 操作 符 
4.4 Foreach 语 法 
4.5 return =e 
4.6 break 和 continue 
4.7 奥 名 昭著 的 goto 
4.8 switch 
49 总 结 … 

第 5 章 初始 化 与 清理 
5.1 用 构造 器 确保 初始 化 
5.2 方法 重 载 ……… 

5.2.1 区 分 重 载 方法 
5.2.2 涉及 基本 类 型 的 重 载 
5.2.3 以 返回 值 区 分 重 载 方 
53 默认 构造 器 …………… 
5.4 this 关 键 字 
5.4.1 在 构造 器 中 调用 构造 器 
5.4.2 static 的 含义 
5.5 清理 : 终结 处 理 和 垃圾 回收 
5.5.1 finalize() 的 用 途 何在 … 
5.5.2 你 必须 实施 清理 
5.5.3 终结 条 件 …… 
5.5.4 垃圾 回收 器 如 何 工作 
5.6 成 员 初 始 化 …… 
5.6.1 指定 初始 化 


























5.7 构造 器 初始 化 * 
5.7.1 初始 化 顺序 
5.7.2 静态 数据 的 初始 人 
573 显 式 的 静态 初始 化 … 
5.7.4 非 静态 实例 初始 人 

5.8 数组 初始 化 ……… 
5.8.1 可 变 参数 列表 

5.9 枚 举 类 型 

5.10 总 结 

第 6 章 访问 权限 控制 

6.1 fa: 库 单元 … 
6.1.1 代码 组 织 
6.1.2 创建 独 一 
6.1.3 定制 工具 库 
6.1.4 用 import 改 变 行为 
6.1.5 对 使 用 包 的 忠告 … 

6.2 Java 访 问 权限 修饰 词 
6.2.1 包 访问 权限 … 
6.2.2 public: 接口 访问 权限 
6.2.3 private: 你 无 法 访问 
6.2.4 protected: 继承 访问 权限 

6.3 接口 和 实现 

6.4 类 的 访问 权限 

6.5 总 结 

第 7 章 复 用 类 

7.1 组 合 语法 

7.2 继承 语法 + 
7.2.1 初始 化 基 类 

7.3 代理 

7.4 结合 使 用 组 合 和 继承 
7.4.1 确保 正确 清理 … 
7.4.2 名 称 屏 项 ……… 

7.5 在 组 合 与 继承 之 间 选 择 

7.6 protected 关 键 字 

7.7 向 上 转型 … 
7.7.1 为 什么 称 为 向 上 转型 
7.7.2 再 论 组 合 与 继承 

7.8 final 关 键 字 
7.8.1 final 数据 
7.8.2 final 方法 
7.8.3 final & … 
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7.8.4 有 关 final 的 忠告 ………………145 
7.9 初始 化 及 类 的 加 载 
7.9.1 继承 与 初始 化 
7.10 总 结 
8.! 再 论 向 上 转型 
8.1.1 忘记 对 象 类 型 
8.2 转机 
8.2.1 方法 调用 绑 定 
8.2.2 产生 正确 的 行为 
8.2.3 
8.2.4 
8.2.5 























缺陷 :“ 覆 盖 ” 私 有 方法 
缺陷 ， 域 与 静态 方法 


8.3 构造 器 和 多 态 ………… 
8.3.1 构造 器 的 调用 顺序 
8.3.2 继承 与 清理 …… 
8.3.3 构造 器 内 部 的 多 态 方法 的 行为 
8.4 协 变 返回 类 型 … 
8.5 用 继承 进行 设计 
8.5.1 纯 继承 与 扩展 pp 





9.1 抽象 类 和 抽象 方法 
9.2 接口 ， 


9.4 Java 中 的 多 重 继承 
9.5 通过 继承 来 扩展 接口 
9.5.1 组 合 接口 时 的 名 字 冲 突 
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9.6 适 配 接口 ， “181 
9.7 接口 中 的 域 +183 
9.7.1 初始 化 接口 中 的 域 

9.8 WERO 


99 接口 与 工厂 … 


9.10 总 结 “188 
第 10 章 内 部 类 -190 
10.1 创建 内 部 类 … j 





10.2 链接 到 外 部 类 “ 
10.3 使 用 .this 与 .new 
10.4 内 部 类 与 向 上 转型 
10.5 在 方法 和 作用 域内 的 内 部 类 








+194 


10.6 匿名 内 部 类 ……… 
10.6.1 再 访 工厂 方法 
10.7 ER 
10.7.1 接口 内 部 的 类 
10.7.2 MSR 
10.8 为 什么 需要 内 部 类 
10.8.1 闭 包 与 回调 … 
10.8.2 内 部 类 与 控制 框架 
10.9 内 部 类 的 继承 …… 
10.10 内 部 类 可 以 被 覆盖 吗 
10.11 局 部 内 部 类 …… 
10.12 内 部 类 标识 符 
10.13 总 结 … 

第 ll 章 持 有 对 象 
11.1 泛 型 和 类 型 安全 的 容器 
11.2 基本 概念 … 
11.3 添加 一 组 元 素 
11.4 容器 的 打印 
11.5 List … 
11.6 和 迭代 器 …… 
11.6.1 Listlterator 
11.7 LinkedList … 
11.8 Stack 
11.9 Set 
11.10 Map 
11.11 Queue 
11.11.1 PriorityQueue 
11.12 Collection 和 Tterator 
11.13 Foreach 与 选 代 器 …… 
11.13.1 halle 
11.14 Be + 

第 12 章 Pervert 
121 概念 
12.2 基本 异常 
12.2.1 异常 参数 
12.3 捕获 异常 
12.3.1 try 块 
12.3.2 异常 处 理 程序 
12.4 创建 自 定义 异常 
12.4.1 异常 与 记录 日 
12.5 异常 说 明 





































12.6 捕获 所 有 异常 “256 
12.6.1 栈 轨迹 … 257 
12.6.2 重新 抛 出 异常 "258 
12.6.3 异常 链 … “260 





+263 
+263 
+264 
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+267 


12.7 Java 标 准 异 常 
12.7.1 特例 ， RuntimeException 
12.8 使 用 finally 进 行 清理 
12.8.1 finally 用 来 做 什么 
12.8.2 在 return 中 使 用 finally 














12.8.3 tht: 异常 丢失 "268 
12.9 异常 的 限制 …… “269 
12.10 构造 器 … -271 
12.11 异常 匹配 





12.12 其 他 可 选 方式 
12.12.1 历史 …… 
12.12.2 观点 
12.12.3 把 异常 传递 给 控制 台 
12.12.4 把 “被 检查 的 异常 ”转换 为 

“不 检查 的 异常 ” 

12.13 异常 使 用 指南 

12.14 总 结 … 

第 13 章 PHE 

13.1 不 可 变 String eee 

13.2 重 载 “+” 与 StringBuilder 

13.3 无 意识 的 递归 … 

13.4 String 上 的 操作 

13.5 格式 化 输出 
13.5.1 printf() 
13.5.2 System.out.format() 
13.5.3 Formatter% + 
13.5.4 格式 化 说 明 符 
13.5.5 Formatter 转 换 
13.5.6 String.format() 

13.6 正则 表达 式 
13.6.1 
13.6.2 
13.6.3 
13.6.4 Pattern 和 Matcher 
13.6.5 split() eee 
13.6.6 替换 操作 
13.6.7 reset() 
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13.6.8 正则 表达 式 与 Java I 
13.7 扫描 输入 
13.7.1 Scanner 定 界 符 + 
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第 1 章 对 象 导 论 


“我 们 之 所 以 将 自然 界 分 解 ， 组 织 成 各 种 概念 ， 并 按 其 含义 分 类 ， 主 要 是 因为 我 们 是 整个 口 
语 交 流 社会 共同 遵守 的 协定 的 参与 者 ， 这 个 协定 以 语言 的 形式 固定 下 来 …… 除 非 赞 成 这 个 协定 
中 规定 的 有 关 语言 信息 的 组 织 和 分 类 ， 否 则 我 们 根本 无 法 交谈 。 

—Benjamin Lee Whorf (1897 ~ 1941) 
计算 机 革命 起 源 于 机 器 ， 因 此 ， 编 程 语言 的 产生 也 始 于 对 机 器 的 模仿 。 

但 是 ， 计 算 机 并 非 只 是 机 器 那么 简单 。 计 算 机 是 头脑 延伸 的 工具 (就 像 Steve Jobs 常 喜欢 说 
的 “头脑 的 自行 车 ”一 样 )， 同 时 还 是 一 种 不 同类 型 的 表达 媒体 。 因 此 ， 这 种 工具 看 起 来 已 经 越 
来 越 不 像 机 器 ， 而 更 像 我 们 头脑 的 一 部 分 ， 以 及 一 种 如 写作 、 绘 画 、 雕 刻 、 动 画 、 电 影 等 一 样 
的 表达 形式 。 面 向 对 象 程序 设计 (Object-oriented Programming, OOP) 便 是 这 种 以 计算 机 作为 表 
达 媒 体 的 大 趋势 中 的 组 成 部 分 。 

本 章 将 向 读者 介绍 包括 开发 方法 概述 在 内 的 OOP 的 基本 概念 。 本 章 ， 乃 至 本 书 中 ， 都 假设 
读者 已 经 具备 了 某 些 编程 经 验 (当然 不 一 定 是 C 的 )。 如 果 读 者 认为 在 阅读 本 书 之 前 还 需要 在 程 
序 设 计 方面 多 做 些 准备 , 那么 就 应 该 去 研读 可 以 从 www.MindView.net 网 站 上 下 载 的 《C 编 程 思想 》 
(Thinking inC) 的 多 媒体 资料 。 

本 章 介绍 的 是 背景 性 的 和 补充 性 的 材料 。 许 多 人 在 没有 了 解 面向 对 象 程序 设计 的 全 哎 之 前 ， 
感觉 无 法 轻松 自在 地 从 事 此 类 编程 。 因 此 ， 此 处 将 引入 许多 概念 ， 以 期 帮助 读者 扎实 地 了 解 
OOP, Rii, 还 有 些 人 可 能 在 看 到 具体 结构 之 前 ， 无 法 了 解 面向 对 象 程序 设计 的 全 貌 ， 这 些 人 
如 果 没 有 代码 在 手 ， 就 会 陷于 困境 并 最 终 迷 失 方 向 。 如 果 你 属于 后 面 这 个 群体 ， 并 且 渴 望 尽快 
获取 Java 语 言 的 细节 ， 那 么 可 以 先 越过 本 章 一 一 在 此 处 越过 本 章 并 不 会 妨碍 你 编写 程序 和 学 习 语 
言 。 但 是 ， 你 最 终 还 是 要 回 到 本 章 来 补充 所 学 知识 ， 这 样 才能 够 了 解 到 对 象 的 重要 性 ， 以 及 怎 
样 使 用 对 象 进行 设计 。 


1.1 抽象 过 程 
所 有 编程 语言 都 提供 抽象 机 制 。 可 以 认为 ， 人 们 所 能 够 解决 的 问题 的 复杂 性 直接 取决 于 抽 


象 的 类 型 和 质量 。 所 谓 的 “类 型 ”是 指 “ 所 抽象 的 是 什么 ?》 ”汇编 语言 是 对 底层 机 器 的 轻微 抽 


象 。 接 着 出 现 的 许多 所 谓 “ 命 令 式 ”语言 (如 FORTRAN、BASIC、C 等 ) 都 是 对 汇编 语言 的 抽 
象 。 这 些 语 言 在 汇编 语言 基础 上 有 了 大 幅 的 改进 ， 但 是 它们 所 作 的 主要 抽象 仍 要 求 在 解决 问题 
时 要 基于 计算 机 的 结构 ， 而 不 是 基于 所 要 解决 的 问题 的 结构 来 考虑 。 程 序 员 必须 建立 起 在 机 器 
模型 (位 于 “ 解 空间 ”内 ， 这 是 你 对 问题 建 模 的 地 方 ， 例 如 计算 机 ) 和 实际 待 解 问题 的 模型 
(位 于 “问题 空间 ”内 ， 这 是 问题 存在 的 地 方 ， 例 如 一 项 业务 ) 之 间 的 关联 。 建 立 这 种 映射 是 费 
力 的 ,而且 这 不 属于 编程 语言 所 固有 的 功能 ， 这 使 得 程序 难以 编写 ， 并 且 维 护 代价 高 吊 ， 同 时 
也 产生 了 作为 副 产 物 的 整个 “编程 方法 ”行业 。 

另 一 种 对 机 器 建 模 的 方式 就 是 只 针对 待 解 问题 建 模 。 早 期 的 编程 语言 ， 如 LISP 和 APL， 都 
选择 考虑 世界 的 某 些 特定 视图 (分 别 对 应 于 “所 有 问题 最 终 都 是 列表 ”或 者 “所 有 问题 都 是 算 
法 形式 的 ")。PROLOG 则 将 所 有 问题 都 转换 成 决策 链 。 此 外 还 产生 了 基于 约束 条 件 编程 的 语言 
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和 专门 通过 对 图 形 符号 操作 来 实现 编程 的 语言 (后 者 被 证 明 限制 性 过 强 )。 这 些 方式 对 于 它们 所 
要 解决 的 特定 类 型 的 问题 都 是 不 错 的 解决 方案 ， 但 是 一 旦 超出 其 特定 领域 ， 它 们 就 力不从心 了 。 

面向 对 象 方式 通过 向 程序 员 提 供 表示 问题 空间 中 的 元 素 的 工具 而 更 进 了 一 步 。 这 种 表示 方 
式 非常 通用 ， 使 得 程序 员 不 会 受 限 于 任何 特定 类 型 的 问题 。 我 们 将 问题 空间 中 的 元 素 及 其 在 解 
空间 中 的 表示 称 为 “对 象 "。( 你 还 需要 一 些 无 法 类 比 为 问题 空间 元 素 的 对 象 。) 这 种 思想 的 实质 
是 : 程序 可 以 通过 添加 新 类 型 的 对 象 使 自身 适用 于 某 个 特定 问题 。 因 此 ， 当 你 在 阅读 描述 解决 
方案 的 代码 的 同时 ， 也 是 在 阅读 问题 的 表述 。 相 比 以 前 我 们 所 使 用 的 语言 ， 这 是 一 种 更 灵活 和 
更 强 有 力 的 语言 抽象 。 所 以 ，OOP 允 许 根据 问题 来 描述 问题 ， 而 不 是 根据 运行 解决 方案 的 计算 
机 来 描述 问题 。 但 是 它 仍然 与 计算 机 有 联系 : 每 个 对 象 看 起 来 都 有 点 像 一 台 微型 计算 机 一 它 具 
有 状态 ， 还 具有 操作 ， 用 户 可 以 要 求 对 象 执行 这 些 操作 。 如 果 要 对 现实 世界 中 的 对 象 作 类 比 ， 
那么 说 它们 都 具有 特性 和 行为 似乎 不 错 。 

Alan Kay 曾 经 总 结 了 第 一 个 成 功 的 面向 对 象 语言 、 同 时 也 是 Java 所 基于 的 语言 之 一 的 
Smalltalk 的 五 个 基本 特性 ， 这 些 特性 表现 了 一 种 纯粹 的 面向 对 象 程序 设计 方式 : 

D 万 物 皆 为 对 象 。 将 对 象 视 为 奇特 的 变量 ， 它 可 以 存储 数据 ， 除 此 之 外 ， 你 还 可 以 要 求 它 
在 自身 上 执行 操作 。 理 论 上 讲 ， 你 可 以 抽取 待 求解 问题 的 任何 概念 化 构件 ( 狗 、 建 筑 物 、 服 务 
等 )， 将 其 表示 为 程序 中 的 对 象 。 

2) 程序 是 对 象 的 集合 ， 它 们 通过 发 送 消息 来 告知 彼此 所 要 做 的 。 要 想 请 求 一 个 对 象 ， 就 必 
须 对 该 对 象 发 送 一 条 消息 。 更 具体 地 说 ， 可 以 把 消息 想像 为 对 某 个 特定 对 象 的 方法 的 调用 请 求 。 

3) 每 个 对 象 都 有 自己 的 由 其 他 对 象 所 构成 的 存储 。 换 句 话说 ， 可 以 通过 创建 包含 现 有 对 象 
的 包 的 方式 来 创建 新 类 型 的 对 象 。 因 此 ， 可 以 在 程序 中 构建 复杂 的 体系 ， 同 时 将 其 复杂 性 隐藏 
在 对 象 的 简单 性 背后 。 

4) 每 个 对 象 都 拥有 其 类 型 。 按 照 通用 的 说 法 ,“ 每 个 对 象 都 是 某 个 类 (class) 的 一 个 实例 
(instance)”， 这 里 “类 ”就 是 “类 型 ”的 同义词 。 每 个 类 最 重要 的 区 别 于 其 他 类 的 特性 就 是 
“可 以 发 送 什么 样 的 消息 给 它 "。 

5) 某 一 特定 类 型 的 所 有 对 象 都 可 以 接收 同样 的 消息 。 这 是 一 句 意味 深长 的 表述 ， 你 在 稍 后 
便 会 看 到 。 因 为 “ 圆 形 ”类 型 的 对 象 同时 也 是 “几何 形 ” 类 型 的 对 象 ， 所 以 一 个 “ 贺 形 ”对 象 
必定 能 够 接受 发 送 给 “几何 形 ”对 象 的 消息 。 这 意味 着 可 以 编写 与 “几何 形 ”交互 并 自动 处 理 
所 有 与 几何 形 性 质 相 关 的 事物 的 代码 。 这 种 可 替代 性 (substitutability) 是 OOP 中 最 强 有 力 的 概 
念 之 一 。 

Booch 对 对 象 提出 了 一 个 更 加 简洁 的 描述 : 对 象 具有 状态 、 行 为 和 标识 。 这 意味 着 每 一 个 对 
象 都 可 以 拥有 内 部 数据 (它们 给 出 了 该 对 象 的 状态 ) 和 方法 它们 产生 行为 )， 并 且 每 一 个 对 象 
都 可 以 唯一 地 与 其 他 对 象 区 分 开 来 ， 具 体 说 来 ， 就 是 每 一 个 对 象 在 内 存 中 都 有 一 个 唯一 的 地 址 ® 。 


1.2 每 个 对 象 都 有 一 个 接口 
亚 里 士 多 德 大 概 是 第 一 个 深入 研究 类 型 (type) 的 哲学 家 ， 他 曾 提出 过 鱼 类 和 鸟 类 这 样 的 概 


念 。 所 有 的 对 象 都 是 唯一 的 ， 但 同时 也 是 具有 相同 的 特性 和 行为 的 对 象 所 归属 的 类 的 一 部 分 。 


O 某 些 编程 语言 的 设计 者 认为 面向 对 象 编程 本 身 不 足以 轻松 地 解决 所 有 编程 问题 ， 所 以 他 们 提倡 将 不 同 的 方式 结 
合 到 多 聚合 编程 语言 (multipleparadigm programming language) 中 。 读 者 可 以 查阅 Timothy Budd 的 
《Multipleparadigm Programming in Leda) 一 书 (Addison-Wesley 1995). 

O 这 确实 显得 有 一 点 过 于 受 限 了 ， 因 为 对 象 可 以 存在 于 不 同 的 机 器 和 地 址 空间 中 ，, 它们 还 可 以 被 存储 在 硬盘 上 。 
在 这 些 情况 下 ， 对 象 的 标识 就 必须 由 内 存 地 址 之 外 的 某 些 东西 来 确定 了 。 
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-这 种 思想 被 直接 应 用 于 第 一 个 面向 对 象 语言 Simula-67， 它 在 程序 中 使 用 基本 关键 字 elass 来 引入 
新 的 类 型 。 

Simula， 就 像 其 名 字 一 样 ， 是 为 了 开发 诸如 经 典 的 “银行 出 纳 员 问 题 ”(bank teller problem) 
这 样 的 仿真 程序 而 创建 的 。 在 银行 出 纳 员 问题 中 ， 有 出 纳 、 客 户 、 账 户 、 交 易 和 货币 单位 等 许 
多 “对 象 "。 在 程序 执行 期 间 具有 不 同 的 状态 而 其 他 方面 都 相似 的 对 象 会 被 分 组 到 对 象 的 类 中 
这 就 是 关键 字 class 的 由 来 。 创 建 抽象 数据 类 型 类 ) 是 面向 对 象 程序 设计 的 基本 概念 之 一 。 抽 
象 数据 类 型 的 运行 方式 与 内 置 《builtin) 类 型 几 平 完全 一 致 :你 可 以 创建 某 一 类 型 的 变量 ( 接 
照 面向 对 象 的 说 法 ， 称 其 为 对 条 或 实例) ， 然 后 操作 这 些 变量 ( 称 其 为 发 送 消息 或 请 求 ， 发 送 消 
息 ， 对 象 就 知道 要 做 什么 )。 每 个 类 的 成 员 或 元 素 都 具有 某 种 共性 : 每 个 账户 都 有 结余 金额 ， 每 
个 出 纳 都 可 以 处 理 存款 请 求 等 。 同 时 ， 每 个 成 员 都 有 其 自身 的 状态 ， 每 个 账户 都 有 不 同 的 结余 
金额 ， 每 个 出 纳 都 有 自己 的 姓名 。 因 此 ， 出 纳 、 客 户 、 账 户 、 交 易 等 都 可 以 在 计算 机 程序 中 被 
表示 成 唯一 的 实体 。 这 些 实体 就 是 对 象 ， 每 一 个 对 象 都 属于 定义 了 特性 和 行为 的 某 个 特定 的 类 。 

所 以 ， 尽 管 我 们 在 面向 对 象 程序 设计 中 实际 上 进行 的 是 创建 新 的 数据 类 型 ， 但 事实 上 所 有 
的 面向 对 象 程序 设计 语言 都 使 用 class 这 个 关键 词 来 表示 数据 类 型 。 当 看 到 类 型 一 词 时 ， 可 将 其 
作为 类 来 考虑 ， 反 之 亦 然 e。 

因为 类 描述 了 具有 相同 特性 (数据 元 素 ) 和 行为 功能) 的 对 象 集合 ， 所 以 一 个 类 实际 上 
就 是 一 个 数据 类 型 ， 例 如 所 有 浮 点 型 数字 具有 相同 的 特性 和 行为 集合 。 二 者 的 差异 在 于 ， 程 序 
员 通过 定义 类 来 适应 问题 ， 而 不 再 被 所 只 能 使 用 现 有 的 用 来 表示 机 器 中 的 存储 单元 的 数据 类 型 。 
可 以 根据 需求 ， 通 过 添加 新 的 数据 类 型 来 扩展 编程 语言 。 编 程 系统 欣然 接受 新 的 类 ， 并 且 像 对 
待 内 置 类 型 一 样 地 照管 它们 和 进行 类 型 检查 。 

面向 对 象 方法 并 不 是 仅 局 限于 构建 仿真 程序 。 无 论 你 是 否 赞成 以 下 观点 ， 即 任何 程序 都 是 
你 所 设计 的 系统 的 一 种 仿真 ， 面 向 对 象 技术 的 应 用 确实 可 以 将 大 量 的 问题 很 容易 地 降解 为 一 个 
简单 的 解决 方案 。 

一 旦 类 被 建立 ， 就 可 以 随心 所 欲 地 创建 类 的 任意 个 对 象 ， 然 后 去 操作 它们 ， 就 像 它们 是 存 
在 于 你 的 待 求解 问题 中 的 元 素 一 样 。 事 实 上 ， 面 向 对 象 程序 设计 的 挑战 之 一 ， 就 是 在 问题 空间 
的 元 素 和 解 空间 的 对 象 之 间 创建 一 对 一 的 映射 。 

但 是 ， 怎 样 才能 获得 有 用 的 对 象 呢 ? 必须 有 某 种 方式 产 
生 对 对 象 的 请 求 ， 使 对 象 完成 各 种 任务 ， 如 完成 一 笔 交易 、 类 型 名 称 
在 屏 葡 上 画图 、 打 开 开关 等 等 。 每 个 对 象 都 只 能 满足 某 些 请 
求 ， 这 些 请 求 由 对 象 的 接口 interface) 所 定义 ， 决 定 接听 的 
便 是 类 型 。 以 电灯 泡 为 例 来 做 一 个 简单 的 比喻 (如 右 图 所 示 ) : 


Light 1t = new Light(); 
tton); 


接口 确定 了 对 某 一 特定 对 象 所 能 发 出 的 请 求 。 但 是 ， 在 程序 中 必须 有 满足 这 些 请 求 的 代码 。 
这 些 代码 与 隐藏 的 数据 一 起 构成 了 实现 。 从 过 程 型 编程 的 观点 来 看 ， 这 并 不 太 复杂 。 在 类 型 中 ， 
每 一 个 可 能 的 请 求 都 有 一 个 方法 与 之 相关 联 ， 当 向 对 象 发 送 请 求 时 ， 与 之 相关 联 的 方法 就 会 被 
调用 。 此 过 程 通常 被 概括 为 : 向 某 个 对 象 “ 发 送 消息 ”( 产 生 请 求 ) ， 这 个 对 象 便 知 道 此 消息 的 
目的 ， 然 后 执行 对 应 的 程序 代码 。 

上 例 中 ， 类 型 /类 的 名 称 是 Light， 特 定 的 Light 对 象 的 名 称 是 it， 可 以 向 Light 对 象 发 出 的 请 
RE: 打开 它 、 关 闭 它 、 将 它 调 亮 、 将 它 调 瞳 。 你 以 下 列 方式 创 建 了 一 个 Light 对 象 : 定义 这 个 


O 有 些 人 对 此 会 区 别 对 待 ， 他 们 认为 : 类 型 决定 了 接口 ， 而 类 是 该 接口 的 一 个 特定 实现 。 


ng 
接口 on 


brighten() 
dimO 
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对 象 的 “引用 ”(it) ， 然 后 调用 mew 方 法 来 创建 该 类 型 的 新 对 象 。 为 了 向 对 象 发 送 消息 ， 需 要 声 
明 对 象 的 名 称 ， 并 以 圆 点 符号 连接 一 个 消息 请 求 。 从 预定 义 类 的 用 户 观点 来 看 ， 这 些 差不多 就 
是 用 对 象 来 进行 设计 的 全 部 。 

前 面 的 图 是 UML (Unified Modelling Language， 统 一 建 模 语言 ) 形式 的 图 ， 每 个 类 都 用 一 
个 方 框 表示 ， 类 名 在 方 框 的 顶部 ， 你 所 关心 的 任何 数据 成 员 都 描述 在 方 框 的 中 间 部 分 ， 方 法 
(隶属 于 此 对 象 的 、 用 来 接收 你 发 给 此 对 象 的 消息 的 函数 ) 在 方 框 的 底部 。 通 常 ， 只 有 类 名 和 公 
共 方 法 被 示 于 UML 设 计 图 中 ， 因 此 ， 方 框 的 中 部 就 像 本 例 一 样 并 未 给 出 。 如 果 只 对 类 型 感 兴趣 
那么 方 框 的 底部 甚至 也 不 需要 给 出 。 


1.3 每 个 对 象 都 提供 服务 


当 正 在 试图 开发 或 理解 一 个 程序 设计 时 ， 最 好 的 方法 之 一 就 是 将 对 象 想像 为 “服务 提供 者 "。 
程序 本 身 将 向 用 户 提供 服务 ， 它 将 通过 调用 其 他 对 象 提供 的 服务 来 实现 这 一 目的 。 你 的 目标 就 
是 去 创建 (或 者 最 好 是 在 现 有 代码 库 中 寻找 ) 能 够 提供 理想 的 服务 来 解决 问题 的 一 系列 对 象 。 

着 手 从 事 这 件 事 的 一 种 方式 就 是 问 一 下 自己 :“ 如 果 我 可 以 将 问题 从 表象 中 抽取 出 来 ， 那 么 
什么 样 的 对 象 可 以 马上 解决 我 的 问题 呢 ? ”例如 ， 假 设 你 正在 创建 一 个 夭 记 系统 ， 那 么 可 以 想 
像 ， 系 统 应 该 具有 某 些 包括 了 预定 义 的 籍 记 输 入 屏幕 的 对 象 ， 一 个 执行 籍 记 计算 的 对 象 集合 
以 及 一 个 处 理 在 不 同 的 打印 机 上 打印 支票 和 开发 票 的 对 象 。 也 许 上 述 对 象 中 的 某 些 已 经 存在 了 
但 是 对 于 那些 并 不 存在 的 对 象 ， 它 们 看 起 来 像 什么 样子 ? 它们 能 够 提供 哪些 服务 ? 它们 需要 哪 
些 对 象 才 能 展 行 它们 的 义务 ?如 果 持 续 这 样 做 ， 那 么 最 终 你 会 说 “那个 对 象 看 起 来 很 简单 ， 可 
以 坐 下 来 写 代码 了 ”， 或 者 会 说 “我 肯定 那个 对 象 已 经 存在 了 "。 这 是 将 问题 分 解 为 对 象 集合 的 
一 种 合理 方式 。 

将 对 象 看 作 是 服务 提供 者 还 有 一 个 附带 的 好 处 : 它 有 助 于 提高 对 象 的 内 聚 性 。 高 内 条 是 软件 
设计 的 基本 质量 要 求 之 一 : 这 意味 着 一 个 软件 构件 〈 例 如 一 个 对 象 ， 当 然 它 也 有 可 能 是 指 一 个 方 
法 或 一 个 对 象 库 ) 的 各 个 方面 “组 合 ”得 很 好 。 人 们 在 设计 对 象 时 所 面临 的 一 个 问题 是 ， 将 过 多 
的 功能 都 塞 在 一 个 对 象 中 。 例 如 ， 在 检查 打印 模式 的 模块 中 ， 你 可 以 这 样 设计 一 个 对 象 ， 让 它 
了 解 所 有 的 格式 和 打印 技术 。 你 可 能 会 发 现 ， 这 些 功能 对 于 一 个 对 象 来 说 太 多 了 ， 你 需要 的 是 
三 个 甚至 更 多 个 对 象 ， 其 中 ， 一 个 对 象 可 以 是 所 有 可 能 的 支票 排版 的 目录 ， 它 可 以 被 用 来 查询 
有 关 如 何 打印 一 张 支票 的 信息 ， 另 一 个 对 象 或 对 象 集合 ) 可 以 是 一 个 通用 的 打印 接口 ， 它 知 
道 有 关 所 有 不 同类 型 的 打印 机 的 信息 (但 是 不 包含 任何 有 关 箭 记 的 内 容 ， 它 更 应 该 是 一 个 需要 
购买 而 不 是 自己 编写 的 对 象 ) ， 第 三 个 对 象 通过 调用 另外 两 个 对 象 的 服务 来 完成 打印 任务 。 这 样 
每 个 对 象 都 有 一 个 它 所 能 提供 服务 的 内 育 的 集合 。 在 良好 的 面向 对 象 设计 中 ， 每 个 对 象 都 可 以 很 
好 地 完成 一 项 任务 ， 但 是 它 并 不 试图 做 更 多 的 事 。 就 像 在 这 里 看 到 的 ， 不 仅 允 许 通过 购买 获得 某 
些 对 象 (打印 机 接口 对 象 ) ， 而 且 还 可 以 创建 能 够 在 别处 复 用 的 新 对 象 (支票 排版 目录 对 象 ) 。 

将 对 象 作为 服务 提供 者 看 待 是 一 件 伟大 的 简化 工具 ， 这 不 仅 在 设计 过 程 中 非常 有 用 ， 而 且 
当 其 他 人 试图 理解 你 的 代码 或 重用 某 个 对 象 时 ， 如 果 他 们 看 出 了 这 个 对 象 所 能 提供 的 服务 的 价 
值 ， 它 会 使 调整 对 象 以 适应 其 设计 的 过 程 变 得 简单 得 多 。 


1.4 被 隐藏 的 具体 实现 
将 程序 开发 人 员 按照 角色 分 为 类 创建 者 (那些 创建 新 数据 类 型 的 程序 员 ) Fs PALA Ke 


全 关于 这 个 术语 的 表述 ， 我 感谢 我 的 朋友 Scott Meyers, 
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(那些 在 其 应 用 中 使 用 数据 类 型 的 类 消费 者 ) 是 大 有 神 益 的 。 客 户 端 程序 员 的 目标 是 收集 各 种 用 
来 实现 快速 应 用 开发 的 类 。 类 创建 者 的 目标 是 构建 类 ， 这 种 类 只 向 客户 端 程序 员 暴 露 必需 的 部 
分 ， 而 隐藏 其 他 部 分 。 为 什么 要 这 样 呢 ? 因为 如 果 加 以 隐藏 ， 那 么 客户 端 程 序 员 将 不 能 够 访问 
它 ， 这 意味 着 类 创建 者 可 以 任意 修改 被 隐 蕊 的 部 分 ， 而 不 用 担心 对 其 他 任何 人 造成 影响 。 被 隐 
茂 的 部 分 通常 代表 对 象 内 部 脆弱 的 部 分 ， 它 们 很 容易 被 粗心 的 或 不 知 内 情 的 客户 端 程序 员 所 毁 
坏 ， 因 此 将 实现 隐藏 起 来 可 以 减少 程序 bug。 

在 任何 相互 关系 中 ， 具 有 关系 所 涉及 的 各 方 都 遵守 的 边界 是 十 分 重要 的 事情 。 当 创建 一 个 
类 库 时 ， 就 建立 了 与 客户 端 程序 员 之 间 的 关系 ， 他 们 同样 也 是 程序 员 ， 但 是 他 们 是 使 用 你 的 类 
库 来 构建 应 用 、 或 者 构建 更 大 的 类 库 的 程序 员 。 如 果 所 有 的 类 成 员 对 任何 人 都 是 可 用 的 ， 那 么 
客户 端 程序 员 就 可 以 对 类 做 任何 事情 ， 而 不 受 任何 约束 。 即 使 你 希望 客户 端 程序 员 不 要 直接 操 
作 你 的 类 中 的 某 些 成 员 ， 但 是 如 果 没 有 任何 访问 控制 ， 将 无 法 阻止 此 事 发 生 。 所 有 东西 都 将 赤 
裸 裸 地 暴露 于 世人 面前 。 

因此 ， 访 问 控制 的 第 一 个 存在 原因 就 是 让 客户 端 程序 员 无 法 触及 他 们 不 应 该 触及 的 部 分 一 这 
些 部 分 对 数据 类 型 的 内 部 操作 来 说 是 必需 的 ， 但 并 不 是 用 户 解决 特定 问题 所 需 的 接口 的 一 部 分 。 
这 对 客户 端 程序 员 来 说 其 实 是 一 项 服务 ， 因 为 他 们 可 以 很 容易 地 看 出 哪些 东西 对 他 们 来 说 很 重 
要 ， 而 哪些 东西 可 以 忽略 。 

访问 控制 的 第 二 个 存在 原因 就 是 允许 库 设 计 者 可 以 改变 类 内 部 的 工作 方式 而 不 用 担心 会 影 
响 到 客户 端 程序 员 。 例 如 ， 你 可 能 为 了 减轻 开发 任务 而 以 某 种 简单 的 方式 实现 了 某 个 特定 类 
但 稍 后 发 现 你 必须 改写 它 才 能 使 其 运行 得 更 快 。 如 果 接 口 和 实现 可 以 清晰 地 分 离 并 得 以 保护 
那么 你 就 可 以 轻而易举 地 完成 这 项 工作 。 

Java 用 三 个 关键 字 在 类 的 内 部 设 定 边界 ，public、private、protected。 这 些 访问 指定 词 
(access specifier) 决定 了 紧 跟 其 后 被 定义 的 东西 可 以 被 谁 使 用 。public 表 示 紧 随 其 后 的 元 素 对 任 
何人 都 是 可 用 的 ， 而 private 这 个 关键 字 表 示 除 类 型 创建 者 和 类 型 的 内 部 方法 之 外 的 任何 人 都 不 
能 访问 的 元 素 。private 就 像 你 与 客户 端 程序 员 之 间 的 一 堵 砖 墙 ， 如 果 有 人 试图 访问 private 成 员 ， 
就 会 在 编译 时 得 到 错误 信息 。protected 关 键 字 与 private 作 用 相当 ， 差 别 仅 在 于 继承 的 类 可 以 访 
问 protected 成 员 ， 但 是 不 能 访问 private 成 员 。 稍 后 将 会 对 继承 进行 介绍 

Java 还 有 一 种 默认 的 访问 权限 ， 当 没有 使 用 前 面 提 到 的 任何 访问 指定 词 时 ， 它 将 发 挥 作用 。 
这 种 权限 通常 被 称 为 包 访问 权限 ， 因 为 在 这 种 权限 下 ， 类 可 以 访问 在 同一 个 包 〈 库 构件 ) 中 的 
其 他 类 的 成 员 ， 但 是 在 包 之 外 ， 这 些 成 员 如 同 指定 了 private 一 样 。 


1.5 复 用 具体 实现 


一 旦 类 被 创建 并 被 测试 完 ， 那 么 它 就 应 该 (在 理想 情况 下 ) 代表 一 个 有 用 的 代码 单元 。 事 
实证 明 ， 这 种 复 用 性 并 不 容易 达到 我 们 所 希望 的 那 种 程度 ， 产 生 一 个 可 复 用 的 对 象 设计 需要 丰 
富 的 经 验 和 敏锐 的 洞察 力 。 但 是 一 旦 你 有 了 这 样 的 设计 ， 它 就 可 供 复 用 。 代 码 复 用 是 面向 对 象 
程序 设计 语言 所 提供 的 最 了 不 起 的 优点 之 一 。 

最 简单 地 复 用 某 个 类 的 方式 就 是 直接 使 用 该 类 的 一 个 对 象 ， 此 外 也 可 以 将 那个 类 的 一 个 对 
象 置 于 某 个 新 的 类 中 。 我 们 称 其 为 “创建 一 个 成 员 对 象 "。 新 的 类 可 以 由 任意 数量 、 任 意 类 型 
的 其 他 对 象 以 任意 可 以 实现 新 的 类 中 想 要 的 功能 的 方式 所 组 成 。 因 为 是 在 使 用 现 有 的 类 合成 新 
的 类 ， 所 以 这 种 概念 被 称 为 组 合 (composition) ， 如 果 组 合 是 动态 发 生 的， 那么 它 通常 被 称 为 
聚合 (aggregation)。 组 合 经 常 被 视 为 “has-a”( 拥 有 ) 关系 ， 就 像 我 们 常 说 的 “汽车 拥有 引擎 
一 样 。 
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汽车 | 引擎 


(上 面 这 张 UML 图 用 实心 琴 形 表明 了 组 合 关系 。 我 通常 采用 最 简单 的 形式 : 仅仅 用 一 条 没有 菱形 
的 线 来 表示 关联 。)。 

组 合 带 来 了 极 大 的 灵活 性 。 新 类 的 成 员 对 象 通常 都 被 声明 为 private， 使 得 使 用 新 类 的 客户 
端 程序 员 不 能 访问 它们 。 这 也 使 得 你 可 以 在 不 干扰 现 有 客户 端 代 码 的 情况 下 ， 修 改 这 些 成 员 
也 可 以 在 运行 时 修改 这 些 成 员 对 象 ， 以 实现 动态 修改 程序 的 行为 。 下 面 将 要 讨论 的 继承 并 不 具 
备 这 样 的 灵活 性 ， 因 为 编译 器 必须 对 通过 继承 而 创建 的 类 施加 编译 时 的 限制 。 

由 于 继承 在 面向 对 象 程序 设计 中 如 此 重要 ， 所 以 它 经 常 被 高 度 强调 ， 于 是 程序 员 新 手 就 会 
有 这 样 的 印象 : 处 处 都 应 该 使 用 继承 。 这 会 导致 难以 使 用 并 过 分 复杂 的 设计 。 实 际 上 ， 在 建立 
新 类 时 ， 应 该 首先 考虑 组 合 ， 因 为 它 更 加 简单 灵活 。 如 果 采 用 这 种 方式 ， 设 计 会 变 得 更 加 清晰 。 
一 旦 有 了 一 些 经 验 之 后 ， 便 能 够 看 出 必须 使 用 继承 的 场合 了 


1.6 继承 


对 象 这 种 观念 ， 本 身 就 是 十 分 方便 的 工具 ， 使 得 你 可 以 通过 概念 将 数据 和 功能 封装 到 一 起 ， 
因此 可 以 对 问题 空间 的 观念 给 出 恰当 的 表示 ， 而 不 用 受制 于 必须 使 用 底层 机 器 语言 。 这 些 概念 
用 关键 字 class 来 表示 ， 它 们 形成 了 编程 语言 中 的 基本 单位 。 

遗憾 的 是 ， 这 样 做 还 是 有 很 多 麻烦 在 创建 了 一 个 类 之 后 ， 即 使 另 一 个 新 类 与 其 具有 相似 的 
功能 ， 你 还 是 得 重新 创建 一 个 新 类 。 如 果 我 们 能 够 以 现 有 的 类 为 基础 ， 复 制 
它 ， 然 后 通过 添加 和 修改 这 个 副本 来 创建 新 类 那 就 要 好 多 了 。 通 过 继承 便 可 | | 














以 达到 这 样 的 效果 ， 不 过 也 有 例外 ， 当 源 类 (被 称 为 基 炎 、 起 类 或 父 类 ) 发 
生变 动 时 ， 被 修改 的 “副本 ”( 被 称 为 导出 类 、 继 承 类 或 子 类 ) 也 会 反映 出 这 
些 变动 (如 有 图 所 示 )。 

(这 张 UML 图 中 的 箭头 从 导出 类 指向 基 类 ， 就 像 稍 后 你 会 看 到 的 ， 通 党 
会 存在 一 个 以 上 的 导出 类 。) HR 

类 型 不 仅仅 只 是 描述 了 作用 于 一 个 对 象 集合 上 的 约束 条 件 ， 同 时 还 有 与 
其 他 类 型 之 间 的 关系 。 两 个 类 型 可 以 有 相同 的 特性 和 行为 ， 但 是 其 中 一 个 类 型 可 能 比 另 一 个 含 
有 更 多 的 特性 ， 并 且 可 以 处 理 更 多 的 消息 或 以 不 同 的 方式 来 处 理 消息 )。 继 承 使 用 基 类 型 和 导 
出 类 型 的 概念 表示 了 这 种 类 型 之 间 的 相似 性 。 一 个 基 类 型 包含 其 所 有 导出 类 型 所 共享 的 特性 和 
行为 。 可 以 创建 一 个 基 类 型 来 表示 系统 中 某 些 对 象 的 核心 概念 ， 从 基 类 型 中 导出 其 他 类 型 ， 来 
表示 此 核心 可 以 被 实现 的 各 种 不 同方 式 。 

以 垃圾 回收 机 为 例 ， 它 用 来 归 类 散落 的 垃圾 。“ 垃 圾 ”是 基 类 型 ， 每 一 件 垃圾 都 有 重量 、 价 
值 等 特性 ， 可 以 被 切 碎 、 熔 化 或 分 解 。 在 此 基础 上 ， 可 以 通过 添加 额外 的 特性 (例如 瓶子 有 颜 
E) RIH (PIMOS ABER RK, RAAT URL) 导出 更 具体 的 垃圾 类 型 。 此 外 ， 某 些 行 
为 可 能 不 同 (例如 纸 的 价值 取决 于 其 类 型 和 状态 ) 。 可 以 通过 使 用 继承 来 构建 一 个 类 型 层次 结构 
以 此 来 表示 待 求解 的 某 种 类 型 的 问题 。 

第 二 个 例子 是 经 典 的 几何 形 的 例子 ， 这 在 计算 机 辅助 设计 系统 或 游戏 仿真 系统 中 可 能 被 用 到 


基 类 是 几何 形 ， 每 一 个 几何 形 都 具有 尺寸 、 颜 色 、 位 置 等 ， 同 时 每 一 个 几何 形 都 可 以 被 绘制 、 擦 


O 通常 对 于 大 多 数 图 来 说 ， 这 样 表示 已 经 足够 了 ， 你 并 不 需要 关心 所 使 用 的 是 聚合 还 是 组 合 。 
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除 、 移 动 和 着 色 等 。 在 此 基础 上 ， 可 以 导出 (继承 出 ) 具体 的 几何 形状 一 一 圆 形 、 正 方形 、 三 角 


形 等 一 每 一 种 都 具有 额外 的 特性 和 行为 ， 例 如 某 
些 形状 可 以 被 翻转 。 某 些 行为 可 能 并 不 相同 ， 例 如 
计算 几何 形状 的 面积 。 类 型 层次 结构 同时 体现 了 几 
何 形状 之 间 的 相似 性 和 差异 性 (如 右上 图 所 示 )。 
以 同样 的 术语 将 解决 方案 转换 成 问题 是 大 有 
神 益 的 ， 因 为 不 需要 在 问题 描述 和 解决 方案 描述 之 
间 建 立 许多 中 间 模 型 。 通 过 使 用 对 象 ， 类 型 层次 结 
构成 为 了 主要 模型 ， 因 此 ， 可 以 直接 从 真实 世界 中 
对 系统 的 描述 过 渡 到 用 代码 对 系统 进行 描述 。 事 实 
上 ， 对 使 用 面向 对 象 设计 的 人 们 来 说 ， 困 难 之 一 是 
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从 开始 到 结束 过 于 简单 。 对 于 训练 有 素 、 善 于 寻找 复杂 的 解决 方案 的 头脑 来 说 ， 可 能 会 在 一 开始 


被 这 种 简单 性 给 难 倒 。 


当 继 承 现 有 类 型 时 ， 也 就 创造 了 新 的 类 型 。 这 个 新 的 类 型 不 仅 包括 现 有 类 型 的 所 有 成 员 ( 尽 
管 private 成 员 被 隐藏 了 起 来 ， 并 且 不 可 访问 )， 而 且 更 重要 的 是 它 复制 了 基 类 的 接口 。 也 就 是 说 ， 
所 有 可 以 发 送 给 基 类 对 象 的 消息 同时 也 可 以 发 送 给 导出 类 对 象 。 由 于 通过 发 送 给 类 的 消息 的 类 型 
可 知 类 的 类 型 ， 所 以 这 也 就 意味 着 导出 类 与 基 类 具有 相同 的 类 型 。 在 前 面 的 例子 中 ,“ 一 个 圆 形 也 
就 是 一 个 几何 形 "。 通 过 继承 而 产生 的 类 型 等 价 性 是 理解 面向 对 象 程序 设计 方法 内 涵 的 重要 门槛 。 
由 于 基 类 和 导出 类 具有 相同 的 基础 接口 ， 所 以 伴随 此 接口 的 必定 有 某 些 具体 实现 。 也 就 是 


说 ， 当 对 象 接收 到 特定 消息 时 ， 必 须 有 某 些 代码 
去 执行 。 如 果 只 是 简单 地 继承 一 个 类 而 并 不 做 其 
他 任何 事 ， 那 么 在 基 类 接口 中 的 方法 将 会 直接 继 
承 到 导出 类 中 。 这 意味 着 导出 类 的 对 象 不 仅 与 基 
类 拥有 相同 的 类 型 ， 而 且 还 拥有 相同 的 行为 ， 这 
样 做 没有 什么 特别 意义 。 

有 两 种 方法 可 以 使 基 类 与 导出 类 产生 差异 。 第 
一 种 方法 非常 直接 : 直接 在 导出 类 中 添加 新 方法 。 
这 些 新 方法 并 不 是 基 类 接口 的 一 部 分 。 这 意味 着 基 
类 不 能 直接 满足 你 的 所 有 需求 ， 因 此 必需 添加 更 多 
的 方法 。 这 种 对 继承 简单 而 基本 的 使 用 方式 ， 有 时 
对 问题 来 说 确实 是 一 种 完美 的 解决 方式 。 但 是 ， 应 
该 仔细 考虑 是 否 存在 基 类 也 需要 这 些 额外 方法 的 可 
能 性 。 这 种 设计 的 发 现 与 迭代 过 程 在 面向 对 象 程序 
设计 中 会 经 常 发 生 (如 右 中 图 所 示 )。 

虽然 继承 有 时 可 能 意味 着 在 接口 中 添加 新 方 
法 (尤其 是 在 以 extends 关 键 字 表示 继承 的 Java 中 )， 
但 并 非 总 需 如 此 。 第 二 种 也 是 更 重要 的 一 种 使 导 
出 类 和 基 类 之 间 产 生 差异 的 方法 是 改变 现 有 基 类 
的 方法 的 行为 ， 这 被 称 之 为 礁 差 (overriding) 那 
个 方法 (如 右 下 图 所 示 )。 

要 想 覆 盖 某 个 方法 ， 可 以 直接 在 导出 类 中 创 





Shape 





draw() 
erase() 
move() 
getColor() 
setColor() 

















Circle 








ee | 





Triangle 





Flipvertical() 





FlipHorizontal()| 








Shape 


araw0 
erase(), 
move() 
getColor() 


setColor() 











| 





Circle 


Square 








draw() 
erase() 





draw() 








erase() 





Triangle 
draw() 
erase() 








35 

















36 











37 











Bs] 





8 HLF 





建 该 方法 的 新 定义 即 可 。 你 可 以 说 :“ 此 时 ， 我 正在 使 用 相同 的 接口 方法 ， 但 是 我 想 在 新 类 型 中 
做 些 不 同 的 事情 。” 
1.6.1 “是 一 个 ”与 “ 像 是 一 个 ”关系 

对 于 继承 可 能 会 引发 某 种 争论 : 继承 应 该 只 覆盖 基 类 的 方法 (而 并 不 添加 在 基 类 中 没有 的 
新 方法 ) 吗 ? 如 果 这 样 做， 就 意味 着 导出 类 和 基 类 是 完全 相同 的 类 型 ， 因 为 它们 具有 完全 相同 
的 接口 。 结 果 可 以 用 一 个 导出 类 对 象 来 完全 替代 一 个 基 类 对 象 。 这 可 以 被 视 为 纯粹 葵 代 ， 通 党 
称 之 为 普 代 原则 。 在 某 种 意义 上 ， 这 是 一 种 处 理 继承 的 理想 方式 。 我 们 经 常 将 这 种 情况 下 的 基 
类 与 导出 类 之 间 的 关系 称 为 is-a (是 一 个 ) 关系 ， 因 为 可 以 说 “一 个 圆 形 就 是 一 个 几何 形状 ”。 
判断 是 否 继承 ， 就 是 要 确定 是 否 可 以 用 is-a 来 描述 类 之 间 的 关系 ， 并 使 之 具有 实际 意义 。 

有 了 时 必须 在 导出 类 型 中 添加 新 的 接口 元 素 ， 这样 也 就 扩展 了 接口 。 这 个 新 的 类 型 仍然 可 以 
替代 基 类 ， 但 是 这 种 替代 并 不 完美 ， 因 为 基 类 无 法 访问 新 添加 的 方法 。 这 种 情况 我 们 可 以 描述 
为 is-like-a ( 像 是 一 个 ) 关系 。 新 类 型 具有 旧 类 型 的 接口 ， 但 是 它 还 包含 其 他 方法 ， 所 以 不 能 说 
它们 完全 相同 。 以 空调 为 例 ， 假 设 房子 里 已 经 布线 安装 好 了 所 有 的 冷气 设备 的 控制 器 ， 也 就 是 
说 ， 房 子 具 备 了 让 你 控制 冷气 设备 的 接口 。 想 像 一 下 ， 如 果 空 调 坏 了 ， 你 用 一 个 既 能 制冷 又 能 
制 热 的 热力 泵 替换 了 它 ， 那 么 这 个 热力 泵 就 is-like-a 空 调 ， 但 是 它 可 以 做 更 多 的 事 。 因 为 房子 的 
控制 系统 被 设计 为 只 能 控制 冷气 设备 ， 所 以 它 只 能 和 新 对 象 中 的 制冷 部 分 进行 通信 。 尽 管 新 对 
象 的 接口 已 经 被 扩展 了 ， 但 是 现 有 系统 除了 原来 接口 之 外 ， 对 其 他 东西 一 无 所 知 。 











Thermostat 控制 |cooting system| 
(自动 调 温 器 ) (制冷 系统 ) 
lowerTemperature()| cool() 





he Condition: Heat Pump 
(空调 ) (AR) 


cool) cool() 


heat() 

















当然 ， 在 看 过 这 个 设计 之 后 ， 很 显然 会 发 现 ， 制 冷 系统 这 个 基 类 不 够 一 般 化 ， 应 该 将 其 更 
名 为 “温度 控制 系统 "， 使 其 可 以 包括 制 热 功能 ， 这 样 我 们 就 可 以 套用 替代 原则 了 。 这 张 图 说 明 
了 在 真实 世界 中 进行 设计 时 可 能 会 发 生 的 事情 

当 你 看 到 替代 原则 时 ， 很 容易 会 认为 这 种 方式 (纯粹 替代 ) 是 唯一 可 行 的 方式 ， 而 且 事实 
上 ， 用 这 种 方式 设计 是 很 好 的 。 但 是 你 会 时 常 发 现 ， 同 样 显然 的 是 你 必须 在 导出 类 的 接口 中 添 
加 新 方法 。 只 要 仔细 审视 ， 两 种 方法 的 使 用 场合 应 该 是 相当 明显 的 。 


17 伴随 多 态 的 可 互 换 对 象 


在 处 理 类 型 的 层次 结构 时 ， 经 常 想 把 一 个 对 象 不 当 作 它 所 属 的 特定 类 型 来 对 待 ， 而 是 将 其 
当 作 其 基 类 的 对 象 来 对 待 。 这 使 得 人 们 可 以 编写 出 不 依赖 于 特定 类 型 的 代码 。 在 “几何 形 ” 的 
例子 中 ， 方 法 操作 的 都 是 证 化 (generic) 的 形状 ， 而 不 关心 它们 是 圆 形 、 正 方形 、 三 角形 还 是 
其 他 什么 尚未 定义 的 形状 。 所 有 的 几何 形状 都 可 以 被 绘制 、 擦 除 和 移动 ， 所 以 这 些 方法 都 是 直 
接 对 一 个 几何 形 对 象 发 送 消息 ， 它 们 不 用 担心 对 象 将 如 何 处 理 消息 。 

这 样 的 代码 是 不 会 受 添加 新 类 型 影响 的 ， 而 且 添加 新 类 型 是 扩展 一 个 面向 对 象 程序 以 便 处 
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理 新 情况 的 最 常用 方式 。 例 如 ， 可 以 从 “几何 形 ” 中 导出 一 个 新 的 子 类 型 “五 角形 ”， 而 并 不 需 
要 修改 处 理 泛 化 几何 形状 的 方法 。 通 过 导出 新 的 子 类 型 而 轻松 扩展 设计 的 能 力 是 对 改动 进行 封 
装 的 基本 方式 之 一 。 这 种 能 力 可 以 极 大 地 改善 我 们 的 设计 ， 同 时 也 降低 软件 维护 的 代价 。 

但 是 ， 在 试图 将 导出 类 型 的 对 象 当 作 其 泛 化 基 类 型 对 象 来 看 待 时 (把 贺 形 看 作 是 几何 形 ， 
把 自行 车 看 作 是 交通 工具 ， 把 牌 屿 看 作 是 乌 等 等 )， 仍 然 存在 一 个 问题 。 如 果 某 个 方法 要 让 泛 化 
几何 形状 绘制 自己 、 让 泛 化 交通 工具 行驶 ， 或 者 让 泛 化 的 鸟 类 移动 ， 那 么 编译 器 在 编译 时 是 不 
可 能 知道 应 该 执行 哪 一 段 代码 的 。 这 就 是 关键 所 在 ， 当 发 送 这 样 的 消息 时 ， 程 序 员 并 不 想 知道 
哪 一 段 代 码 将 被 执行 ， 绘图 方法 可 以 被 等 同 地 应 用 于 圆 形 、 正 方形 、 三 角形 ， 而 对 象 会 依据 自 
身 的 具体 类 型 来 执行 恰当 的 代码 。 

如 果 不 需 要 知道 哪 一 段 代码 会 被 执行 ， 那 么 当 添 加 新 的 子 类 型 时 ， 不 需要 更 改 调用 它 的 方 
法 ， 它 就 能 够 执行 不 同 的 代码 。 因 此 ， 编 译 器 无 法 精确 地 了 解 哪 一 段 代码 将 会 被 执行 ， 那 么 它 
该 怎么 办 呢 ? 例如 ， 在 下 面 的 图 中 ，BirdController 对 象 仅仅 处 理 泛 化 的 Bird 对 象 ， 而 不 了 解 它 
们 的 确切 类 型 。 从 BirdController 的 角度 看 ， 这 么 做 非常 方便 ， 不 需要 编写 特别 的 代码 来 判 
定 要 处 理 的 Bird 对 象 的 确切 类 型 或 其 行为 。 当 move0 方 法 被 调用 时 ， 即 便 忽 略 Bird 的 具体 类 型 ， 
也 会 产生 正确 的 行为 《Goose (5) 走 、 飞 或 游泳 ，Penguin (£H) 走 或 游泳 ) ， 那 么 ， 这 是 如 


何 发 生 的 呢 ? 
lsirdcontrotter| Bird 
当 调用 move () 时 会 
reLocate() 发 生 什么 ? move() 
Goose Penguin 
move() movel 


这 个 问题 的 答案 ， 也 是 面向 对 象 程序 设计 的 最 重要 的 妙 诀 : 编译 器 不 可 能 产生 传统 意义 上 
的 函数 调用 。 一 个 非 面 向 对 象 编程 的 编译 器 产生 的 函数 调用 会 引起 所 谓 的 前 期 绑 定 ， 这 个 术语 
你 可 能 以 前 从 未 听 说 过 ， 可 能 从 未 想 过 函数 调用 的 其 他 方式 。 这 么 做 意味 着 编译 器 将 产生 对 一 
个 具体 函数 名 字 的 调用 ， 而 运行 时 将 这 个 调用 解析 到 将 要 被 执行 的 代码 的 绝对 地 址 。 然 而 在 
OOP 中 ， 程 序 直 到 运行 时 才能 够 确定 代码 的 地 址 ， 所 以 当 消 息 发 送 到 一 个 泛 化 对 象 时 ， 必 须 采 
用 其 他 的 机 制 。 

为 了 解决 这 个 问题 , 面向 对 象 程序 设计 语言 使 用 了 后 期 绑 定 的 概念 。 当 向 对 象 发 送 消 息 时 ， 
被 调用 的 代码 直到 运行 时 才能 确定 。 编 译 器 确保 被 调用 方法 的 存在 ， 并 对 调用 参数 和 返回 值 
执行 类 型 检查 〈 无 法 提供 此 类 保证 的 语言 被 称 为 是 弱 类 型 的 ) ， 但 是 并 不 知道 将 被 执行 的 确切 
代码 。 

为 了 执行 后 期 绑 定 ，Java 使 用 一 小 段 特殊 的 代码 来 替代 绝对 地 址 调用 。 这 段 代码 使 用 在 对 
象 中 存储 的 信息 来 计算 方法 体 的 地 址 (这 个 过 程 将 在 第 8 章 中 详 述 )。 这 样 ， 根 据 这 一 小 段 代 码 
的 内 容 ， 每 一 个 对 象 都 可 以 具有 不 同 的 行为 表现 。 当 向 一 个 对 象 发 送 消息 时 ， 该 对 象 就 能 够 知 
道 对 这 条 消息 应 该 做 些 什么 。 ` 

在 某 些 语 言 中 ， 必 须 明 确 地 声明 希望 某 个 方法 具备 后 期 绑 定 属性 所 带 来 的 灵活 性 〈C++ 是 
使 用 virtual 关 键 字 来 实现 的 )。 在 这 些 语言 中 ， 方 法 在 默认 情况 下 不 是 动态 绑 定 的 。 而 在 Java 中 ， 
动态 绑 定 是 默认 行为 ， 不 需要 添加 额外 的 关键 字 来 实现 多 态 。 
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再 来 看 看 几何 形状 的 例子 。 整 个 类 族 (其 中 所 有 的 类 都 基于 相同 的 一 致 接口 ) 在 本 章 前 面 
已 有 图 示 。 为 了 说 明 多 态 ， 我 们 要 编写 一 段 代 码 ， 它 忽略 类 型 的 具体 细节 ， 仅 仅 和 基 类 交互 。 
这 段 代码 和 具体 类 型 信息 是 分 离 的 (decoupled)， 这 样 做 使 代码 编写 更 为 简单 ， 也 更 易于 理解 。 
而 且 ， 如 果 通 过 继承 机 制 添加 一 个 新 类 型 ， 例 如 Hexagon (六 边 形 )， 所 编写 的 代码 对 Shape 
(几何 形 ) 的 新 类 型 的 处 理 与 对 已 有 类 型 的 处 理会 同样 出 色 。 正 因为 如 此 ， 可 以 称 这 个 程序 是 可 
扩展 的 。 

如 果 用 Java 来 编写 一 个 方法 (后 面 很 快 你 就 会 学 习 如 何 编写 ) : 


void doSomething(Shape shape) { 
shape.erase(); 
ts 
shape.draw(); 


这 个 方法 可 以 与 任何 Shape 对 话 , 因此 它 是 独立 于 任何 它 要 绘制 和 控 除 的 对 象 的 具体 类 型 的 。 
如 果 程 序 中 其 他 部 分 用 到 了 doSomething( 方 法 ; 


Circle circle = new Circle(); 
Triangle triangle = new Triangle(); 
Line line = new Line(); 
doSomething(circle); 
doSomething(triangle) ; 
doSomething(Line) ; 


对 doSomething0 的 调用 会 自动 地 正确 处 理 ， 而 不 管 对 象 的 确切 类 型 。 

这 是 一 个 相当 令 人 惊奇 的 诀窍。 看 看 下 面 这 行 代码 ， 

doSomething(circle); 

当 Cirelke 被 传人 到 预期 接收 Shape 的 方法 中 ， 究 竟 会 发 生 什么 。 由 于 Cirdle 可 以 被 doSomething0 
看 作 是 Shape， 也 就 是 说 ，doSomething0 可 以 发 送 给 Shape 的 任何 消息 ，Circle 都 可 以 接收 ， 那 
么 ， 这 么 做 是 完全 安全 且 合 平 逻辑 的 。 

把 将 导出 类 看 做 是 它 的 基 类 的 过 程 称 为 向 上 转型 (upcasting)。 转 型 (cast) 这 个 名 称 的 灵 
感 来 自 于 模型 铸造 的 塑 模 动作 ， 而 向 上 (up) 这 
个 词 来 源 于 继承 图 的 册 型 布局 方式 ,通常 基 类 在 人 | shape 
顶部 ， 而 导出 类 在 其 下 部 散 开 。 因 此 ， 转 型 为 一 “向 上 转型 | 
个 基 类 就 是 在 继承 图 中 向 上 移动 ， 即 “向 上 转型 f 
(如 右 图 所 示 )。 pi 

一 个 面向 对 象 程序 肯定 会 在 某 处 包含 向 上 转 :| arae ] TS 
型 ， 因 为 这 正 是 将 自己 从 必须 知道 确切 类 型 中 解 
放出 来 的 关键 。 让 我 们 再 看 看 doSomething0 中 的 代码 。 


shape.erase(); 
LP hes 












































shape.draw(); 

注意 这 些 代码 并 不 是 说 “如 果 是 Circle， 请 这 样 做 ， 如 果 是 Square， 请 那样 做 ……”。 如 果 
编写 了 那 种 检查 Shape 所 有 实际 可 能 类 型 的 代码 ， 那 么 这 段 代码 肯定 是 杂乱 不 堪 的 ， 而 且 在 每 次 
添加 了 Shape 的 新 类 型 之 后 都 要 去 修改 这 段 代 码 。 这 里 所 要 表达 的 意思 仅仅 是 “你 是 一 个 Shape， 
我 知道 你 可 以 erase0 和 draw0 你 自己 ， 那 么 去 做 吧 ， 但 是 要 注意 细节 的 正确 性 .” 

doSomething0 的 代码 给 人 印象 深刻 之 处 在 于 ， 不 知 何故 ， 它 总 是 做 了 该 做 的 。 调 用 Circle 
的 draw0 方 法 所 执行 的 代码 与 调用 Square 或 Line 的 draw0 方 法 所 执行 的 代码 是 不 同 的 ， 而 且 当 
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draw0 消 息 被 发 送 给 一 个 匿名 的 Shape 时 ， 也 会 基于 该 Shape 的 实际 类 型 产生 正确 的 行为 。 这 相 
当 神 奇 ， 因 为 就 像 在 前 面 提 到 的 ， 当 Java 编 译 器 在 编译 doSomething() 的 代码 时 ， 并 不 能 确切 知 
道 doSomething0 要 处 理 的 确切 类 型 。 所 以 通常 会 期 望 它 的 编译 结果 是 调用 基 类 Shape 的 erase( 
和 draw0 版 本 ， 而 不 是 具体 的 Circle、Square 或 Line 的 相应 版 本 。 正 是 因为 多 态 才 使 得 事情 总 是 
能 够 被 正确 处 理 。 编 译 器 和 运行 系统 会 处 理 相关 的 细节 ， 你 需要 马上 知道 的 只 是 事情 会 发 生 ， 
更 重要 的 是 怎样 通过 它 来 设计 。 当 向 一 个 对 象 发 送 消息 时 ， 即 使 涉及 向 上 转型 ， 该 对 象 也 知道 
要 执行 什么 样 的 正确 行为 。 


1.8 单 根 继承 结构 


在 OOP 中 ， 自 C++ 面世 以 来 就 已 变 得 非常 眠 目的 一 个 问题 就 是 ， 是 否 所 有 的 类 最 终 都 继承 
自 单一 的 基 类 。 在 Java 中 事实 上 还 包括 除 C++ 以 外 的 所 有 OOP 语 言 )， 答 案 是 yes， 这 个 终极 基 
类 的 名 字 就 是 Object。 事 实证 明 ， 单 根 继承 结构 带 来 了 很 多 好 处 。 

在 单 根 继承 结构 中 的 所 有 对 象 都 具有 一 个 共用 接口 ， 所 以 它们 归根 到 底 都 是 相同 的 基本 类 
型 。 另 一 种 (C++ 所 提供 的 ) 结构 是 无 法 确保 所 有 对 象 都 属于 同一 个 基本 类 型 。 从 向 后 兼容 的 
角度 看 ， 这 么 做 能 够 更 好 地 适应 C 模 型 ， 而 且 受 限 较 少 ， 但 是 当 要 进行 完全 的 面向 对 象 程序 设计 
时 ， 则 必须 构建 自己 的 继承 体系 ， 使 得 它 可 以 提供 其 他 OOP 语 言 内 置 的 便利 。 并 且 在 所 获得 的 
任何 新 类 库 中 ， 总 会 用 到 一 些 不 兼容 的 接口 ， 需 要 花 力气 (有 可 能 要 通过 多 重 继承 ) 来 使 新 接 
口 融入 你 的 设计 之 中 。 这 么 做 来 换取 Ci+ 额 外 的 灵活 性 是 否 值得 呢 ? 如 果 需 要 的 话 一 如 果 在 C 
上 面 投资 巨大 ， 这 么 做 就 很 有 价值 。 如 果 是 刚刚 从 头 开始 ， 那 么 像 Java 这 样 的 选择 通常 会 有 更 
高 的 生产 率 。 

单 根 继承 结构 保证 所 有 对 象 都 具备 某 些 功能 。 因 此 你 知道 ， 在 你 的 系统 中 你 可 以 在 每 个 对 象 
上 执行 某 些 基本 操作 。 所 有 对 象 都 可 以 很 容易 地 在 堆 上 创建 ， 而 参数 传递 也 得 到 了 极 大 的 简化 。 

单 根 继承 结构 使 垃圾 回收 器 的 实现 变 得 容易 得 多 ， 而 垃圾 回收 器 正 是 Java 相 对 C++ 的 重要 改 
进 之 一 。 由 于 所 有 对 象 都 保证 具有 其 类 型 信息 ， 因 此 不 会 因 无 法 确定 对 象 的 类 型 而 陷 人 僵局 。 
这 对 于 系统 级 操作 (如 异常 处 理 ) 显得 尤其 重要 ， 并 且 给 编程 带 来 了 更 大 的 灵活 性 。 


1.9 容器 


通常 说 来 ， 如 果 不 知 道 在 解决 某 个 特定 问题 时 需要 多 少 个 对 象 ， 或 者 它们 将 存活 多 久 ， 那 
么 就 不 可 能 知道 如 何 存储 这 些 对 象 。 如 何 才能 知道 需要 多 少 空间 来 创建 这 些 对 象 呢 ? 答案 是 你 
不 可 能 知道 ， 因 为 这 类 信息 只 有 在 运行 时 才能 获得 。 

对 于 面向 对 象 设计 中 的 大 多 数 问题 而 言 ， 这 个 问题 的 解决 方案 似乎 过 于 轻率 ， 创建 另 一 种 
对 象 类 型 。 这 种 新 的 对 象 类 型 持 有 对 其 他 对 象 的 引用 。 当 然 ， 你 可 以 用 在 大 多 数 语言 中 都 有 的 
数组 类 型 来 实现 相同 的 功能 。 但 是 这 个 通常 被 称 为 容器 (也 称 为 集合 ， 不 过 Java 类 库 以 不 同 的 
含义 使 用 “集合 ”这 个 术语 ， 所 以 本 书 将 使 用 “容器 ”这 个 词 ) 的 新 对 象 ， 在 任何 需要 时 都 可 
扩充 自己 以 容纳 你 置 于 其 中 的 所 有 东西 。 因 此 不 需要 知道 将 来 会 把 多 少 个 对 象 置 于 容器 中 ， 只 
需要 创建 一 个 容器 对 象 ， 然 后 让 它 处 理 所 有 细节 。 

幸运 的 是 ， 好 的 OOP 语 言 都 有 一 组 容器 ， 它 们 作为 开发 包 的 一 部 分 。 在 C++ 中 ， 容 器 是 标 
准 C++ 类 库 的 一 部 分 ， 经 常 被 称 为 标准 模板 类 库 (Standard Template Library, STL), Object Pascal 
在 其 可 视 化 构件 库 (Visual Component Library, VCL) 中 有 容器 ，Smalltalk 提 供 了 一 个 非常 完备 
的 容器 集 ，Java 在 其 标准 类 库 中 也 包含 有 大 量 的 容器 。 在 某 些 类 库 中 ， 一 两 个 通用 容器 足够 满 
足 所 有 的 需要 ， 但 是 在 其 他 类 库 (例如 Java) 中 ， 具 有 满足 不 同 需要 的 各 种 类 型 的 容器 ， 例 如 
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List (用 于 存储 序列 ) Map (也 被 称 为 关联 数组 ， 用 来 建立 对 象 之 间 的 关联 ) Set (每 种 对 象 类 
型 只 持 有 一 个 ) ， 以 及 诸如 队列 、 树 、 堆 栈 等 更 多 的 构件 。 

从 设计 的 观点 来 看 ， 真 正 需要 的 只 是 一 个 可 以 被 操作 ， 从 而 解决 问题 的 序列 。 如 果 单 一 类 
型 的 容器 可 以 满足 所 有 需要 ， 那 么 就 没有 理由 设计 不 同 种 类 的 序列 了 。 然 而 还 是 需要 对 容器 有 
所 选择 ， 这 有 两 个 原因 。 第 一 ， 不 同 容器 提供 了 不 同类 型 的 接口 和 外 部 行为 。 堆 栈 相 比 于 队列 
就 具备 不 同 的 接口 和 行为 ， 也 不 同 于 集合 和 列表 的 接口 和 行为 。 它 们 之 中 的 某 种 容器 提供 的 解 
决 方案 可 能 比 其 他 容器 要 灵活 得 多 。 第 二 ， 不 同 的 容器 对 于 某 些 操作 具有 不 同 的 效率 。 最 好 的 
例子 就 是 两 种 List 的 比较 : ArrayList 和 LinkedList。 它 们 都 是 具有 相同 接口 和 外 部 行为 的 简单 的 
序列 ， 但 是 它们 对 某 些 操作 所 花费 的 代价 却 有 天 壤 之 别 。 在 ArrayList 中 ， 随 机 访问 元 素 是 一 个 
花费 固定 时 间 的 操作 ， 但 是 ， 对 LinkedList 来 说 ， 随 机 选取 元 素 需要 在 列表 中 移动 ， 这 种 代价 是 
高 昂 的 ， 访 问 越 靠 近 表 尾 的 元 素 ， 花 费 的 时 间 越 长 。 而 另 一 方面 ， 如 果 想 在 序列 中 间 插 入 一 个 
元 素 ，LinkedList 的 开销 却 比 ArrayList 要 小 。 上 述 操作 以 及 其 他 操作 的 效率 ， 依 序列 底层 结构 的 
不 同 而 存在 很 大 的 差异 。 我 们 可 以 在 一 开始 使 用 LinkedList 构 建 程序 ， 而 在 优化 系统 性 能 时 改 用 
ArrayList。 接 口 List 所 带 来 的 抽象 ， 把 在 容器 之 间 进 行 转换 时 对 代码 产生 的 影响 降 到 最 小 限度 。 
1.9.1 参数 化 类 型 

在 Java SE5 出 现 之 前 ， 容 器 存储 的 对 象 都 只 具有 Java 中 的 通用 类 型 ， Object。 单 根 继承 结构 
意味 着 所 有 东西 都 是 Object 类 型 ， 所 以 可 以 存储 Object 的 容器 可 以 存储 任何 东西 8 。 这 使 得 容器 
很 容易 被 复 用 。 

要 使 用 这 样 的 容器 ， 只 需 在 其 中 置信 对 象 引 用 ， 稍 后 还 可 以 将 它们 取 回 。 但 是 由 于 容器 只 
存储 Object， 所 以 当 将 对 象 引用 置信 容器 时 ， 它 必须 被 向 上 转型 为 Object， 因 此 它 会 丢失 其 身份 。 
当 把 它 取 回 时 ， 就 获取 了 一 个 对 Object 对 象 的 引用 ， 而 不 是 对 置 入 时 的 那个 类 型 的 对 象 的 引用 。 
所 以 ， 怎 样 才能 将 它 变 回 先前 置 人 容器 中 时 的 具有 实用 接口 的 对 象 呢 ? 

这 里 再 度 用 到 了 转型 ， 但 这 一 次 不 是 向 继承 结构 的 上 层 转型 为 一 个 更 泛 化 的 类 型 ， 而 是 向 
下 转型 为 更 具体 的 类 型 。 这 种 转型 的 方式 称 为 向 下 转型 。 我 们 知道 ， 向 上 转型 是 安全 的 ， 例 如 
Circle 是 一 种 Shape 类 型 ， 但 是 不 知道 某 个 Object 是 Circle 还 是 Shape， 所 以 除非 确切 知道 所 要 处 理 
的 对 象 的 类 型 ， 否 则 向 下 转型 几乎 是 不 安全 的 。 

然而 向 下 转型 并 非 彻底 是 危险 的 如 果 向 下 转型 为 错误 的 类 型 ， 就 会 得 到 被 称 为 异常 
的 运行 时 错误 ， 稍 后 会 介绍 什么 是 异常 。 尽 管 如 此 ， 当 从 容器 中 取出 对 象 引用 时 ， 还 是 必须 要 
以 某 种 方式 记 住 这 些 对 象 究竟 是 什么 类 型 ， 这 样 才 能 执行 正确 的 向 下 转型 。 

向 下 转型 和 运行 时 的 检查 需要 额外 的 程序 运行 时 间 ， 也 需要 程序 员 付出 更 多 的 心血 。 那 么 
创建 这 样 的 容器 ， 它 知道 自己 所 保存 的 对 象 的 类 型 ， 从 而 不 需要 向 下 转型 以 及 消除 犯错 误 的 可 
能 ， 这 样 不 是 更 有 意义 吗 ? 这 种 解决 方案 被 称 为 参数 化 类 型 机 制 。 参 数 化 类 型 就 是 一 个 编译 器 
可 以 自动 定制 作用 于 特定 类 型 上 的 类 。 例 如 ， 通 过 使 用 参数 化 类 型 ， 编 译 器 可 以 定制 一 个 只 接 
纳 和 取出 Shape 对 象 的 容器 。 

Java SE5 的 重大 变化 之 一 就 是 增加 了 参数 化 类 型 ， 在 Java 中 它 称 为 范 型 。 一 对 尖 括 号 ， 中 间 
包含 类 型 信息 ， 通 过 这 些 特征 就 可 以 识别 对 范 型 的 使 用 。 例 如 ， 可 以 用 下 面 这 样 的 语句 来 创建 
一 个 存储 Shape 的 ArrayList: 


ArrayList<Shape> shapes = new ArrayList<Shape>(); 





O 它们 不 能 持 有 基本 类 型 ， 但 是 Java SE5 的 自动 包装 功能 使 得 这 项 限制 几乎 不 成 什么 问题 了 。 有 关 这 一 点 将 在 本 


书后 面 的 章节 中 进行 详细 讨论 。 





It RF : 13 





为 了 利用 范 型 的 优点 ， 很 多 标准 类 库 构 件 都 已 经 进行 了 修改 。 就 像 我 们 将 要 看 到 的 那样 ， 
范 型 对 本 书 中 的 许多 代码 都 产生 了 重要 的 影响 。 


1.10 对 象 的 创建 和 生命 期 


在 使 用 对 象 时 ， 最 关键 的 问题 之 一 便 是 它们 的 生成 和 销毁 方式 。 每 个 对 象 为 了 生存 都 需要 
资源 ， 尤 其 是 内 存 。 当 我 们 不 再 需要 一 个 对 象 时 ， 它 必须 被 清理 掉 ， 使 其 占有 的 资源 可 以 被 释 
放 和 重用 。 在 相对 简单 的 编程 情况 下， 怎样 清理 对 象 看 起 来 似乎 不 是 什么 挑战 ， 你 创建 了 对 象 ， 
根据 需要 使 用 它 ， 然 后 它 应 该 被 销毁 。 然 而 ， 你 很 可 能 会 遇 到 相对 复杂 的 情况 。 

例如 ， 假 设 你 正在 为 某 个 机 场 设计 空中 交通 管理 系统 (同样 的 模型 在 仓库 货柜 管理 系统 、 
录像 带 出 租 系统 或 宠物 寄宿 店 也 适用 )。 一 开始 问题 似乎 很 简单 : 创建 一 个 容器 来 保存 所 有 的 飞 
机 ， 然 后 为 每 一 架 进 入 空中 交通 控制 区 域 的 飞机 创建 一 个 新 的 飞机 对 象 ， 并 将 其 置 于 容器 中 。 
对 于 清理 工作 ， 只 需 在 飞机 离开 此 区 域 时 删除 相关 的 飞机 对 象 即 可 ， 

但 是 ， 可 能 还 有 别 的 系统 记录 着 有 关 飞机 的 数据 ， 也 许 这 些 数据 不 需要 像 主 要 控制 功能 那 
样 立即 引 人 注 意 。 例 如 ， 它 可 能 记录 着 所 有 飞 离 机 场 的 小 型 飞机 的 飞行 计划 。 因 此 你 需要 有 第 
二 个 容器 来 存放 小 型 飞机 ， 无 论 何 时 ， 只 要 创建 的 是 小 型 飞机 对 象 ， 那 么 它 同时 也 应 该 置信 第 
二 个 容器 内 。 然 后 某 个 后 台 进 程 在 空闲 时 对 第 二 个 容器 内 的 对 象 进行 操作 。 

现在 问题 变 得 更 困难 了 : 怎样 才能 知道 何 时 销毁 这 些 对 象 呢 ? 当 处 理 完 某 个 对 象 之 后 ， 系 
统 某 个 其 他 部 分 可 能 还 在 处 理 它 。 在 其 他 许多 场合 中 也 会 过 到 同样 的 问题 ， 在 必须 明确 删除 对 
象 的 编程 系统 中 (例如 C++)， 此 问题 会 变 得 十 分 复杂 。 

对 象 的 数据 位 于 何 处 ?怎样 控制 对 象 的 生命 周期 ? C++ 认为 效率 控制 是 最 重要 的 议题 ， 所 
以 给 程序 员 提供 了 选择 的 权力 。 为 了 追求 最 大 的 执行 速度 ， 对 象 的 存储 空间 和 生命 周期 可 以 在 
编写 程序 时 确定 ， 这 可 以 通过 将 对 象 置 于 堆栈 (它们 有 时 被 称 为 自动 变量 (automatic variable) 
或 限 域 变量 (scoped variable)) 或 静态 存储 区 域内 来 实现 。 这 种 方式 将 存储 空间 分 配 和 释放 置 于 
优先 考虑 的 位 置 ， 某 些 情况 下 这 样 控制 非常 有 价值 。 但 是 ， 也 辆 牲 了 灵活 性 ， 因 为 必须 在 编写 
程序 时 知道 对 象 确切 的 数量 、 生 命 周 期 和 类 型 。 如 果 试 图 解决 更 一 般 化 的 问题 ， 例 如 计算 机 辅 
助 设计 、 仓 库 管 理 或 者 空中 交通 控制 ， 这 种 方式 就 显得 过 于 受 限 了 。 

第 二 种 方式 是 在 被 称 为 堆 (heap) 的 内 存 池 中 动态 地 创建 对 象 。 在 这 种 方式 中 ， 直 到 运行 
时 才 知 道 需 要 多 少 对 象 ， 它 们 的 生命 周期 如 何 ， 以 及 它们 的 具体 类 型 是 什么 。 这 些 问题 的 答案 
只 能 在 程序 运行 时 相关 代码 被 执行 到 的 那 一 刻 才能 确定 。 如 果 需 要 一 个 新 对 象 ， 可 以 在 需要 的 
时 刻 直接 在 堆 中 创建 。 因 为 存储 空间 是 在 运行 时 被 动态 管理 的 ， 所 以 需要 大 量 的 时 间 在 堆 中 分 
配 存储 空间 ， 这 可 能 要 远 远大 于 在 堆栈 中 创建 存储 空间 的 时 间 。 在 堆栈 中 创建 存储 空间 和 释放 
存储 空间 通常 各 需要 一 条 汇编 指令 即 可 ， 分 别 对 应 将 栈 顶 指针 向 下 移动 和 将 栈 顶 指针 向 上 移动 。 
创建 堆 存 储 空间 的 时 间 依赖 于 存储 机 制 的 设计 。 

动态 方式 有 这 样 一 个 一 般 性 的 逻辑 假设 : 对 象 趋向 于 变 得 复杂 ， 所 以 查找 和 释放 存储 空间 
的 开销 不 会 对 对 象 的 创建 造成 重大 冲击 。 动 态 方式 所 带 来 的 更 大 的 灵活 性 正 是 解决 一 般 化 编程 
问题 的 要 点 所 在 。 

Java 完 全 采用 了 动态 内 存 分 配方 式 ” 。 每 当 想 要 创建 新 对 象 时 ， 就 要 使 用 new 关 键 字 来 构建 
此 对 象 的 动态 实例 。 

还 有 一 个 议题 ， 就 是 对 象 生 命 周期 。 对 于 人 允许 在 堆栈 上 创建 对 象 的 语言 ， 编 译 器 可 以 确定 


O 稍 候 你 将 看 到 ， 基 本 类 型 只 是 一 各 特例 。 
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对 象 存活 的 时 间 ， 并 可 以 自动 销毁 它 。 然 而 ， 如 果 是 在 堆 上 创建 对 象 ， 编 译 器 就 会 对 它 的 生命 
周期 一 无 所 知 。 在 像 C++ 这 样 的 语言 中 ， 必 须 通过 编程 方式 来 确定 何 时 销毁 对 象 ， 这 可 能 会 因 
为 不 能 正确 处 理 而 导致 内 存 泄漏 (这 在 C++ 程序 中 是 常见 的 问题 ) 。Java 提 供 了 被 称 为 “垃圾 回 
收 器 ”的 机 制 ， 它 可 以 自动 发 现 对 象 何 时 不 再 被 使 用 ， 并 继而 销毁 它 。 垃 圾 回收 器 非常 有 用 ， 
因为 它 减 少 了 所 必须 考虑 的 议题 和 必须 编写 的 代码 。 更 重要 的 是 ， 垃 圾 回收 器 提供 了 更 高 层 的 
保障 ， 可 以 避免 暗藏 的 内 存 泄漏 问题 ， 这 个 问题 已 经 使 许多 C++ 项 目 折 载 沉 沙 。 

Java 的 垃圾 回收 器 被 设计 用 来 处 理 内 存 释放 问题 (尽管 它 不 包括 清理 对 象 的 其 他 方面 )。 垃 
圾 回收 器 “知道 ”对 象 何 时 不 再 被 使 用 ， 并 自动 释放 对 象 占 用 的 内 存 。 这 一 点 同 所 有 对 象 都 是 
继承 自 单 根基 类 Object 以 及 只 能 以 一 种 方式 创建 对 象 (在 堆 上 创建 ) 这 两 个 特性 结合 起 来 ， 使 得 
用 Java 编 程 的 过 程 较 之 用 C++ 编程 要 简单 得 多 ， 所 要 做 出 的 决策 和 要 克服 的 障碍 也 要 少 得 多 。 


1.11 异常 处 理 : 处 理 错误 


自从 编程 语言 问世 以 来 ， 错 误 处 理 就 始终 是 最 困难 的 问题 之 一 。 因 为 设计 一 个 良好 的 错误 
处 理 机 制 非常 困难 ， 所 以 许多 语言 直接 略 去 这 个 问题 ， 将 其 交 给 程序 库 设计 者 处 理 ， 而 这 些 设 
计 者 也 只 是 提出 了 一 些 不 彻底 的 方法 ， 这 些 方法 可 用 于 许多 很 容易 就 可 以 绕 过 此 问题 的 场合 
而 且 其 解决 方式 通常 也 只 是 忽略 此 问题 。 大 多 数 错误 处 理 机 制 的 主要 问题 在 于 ， 它 们 都 依赖 于 
程序 员 自 身 的 警惕 性 ， 这 种 警惕 性 来 源 于 一 种 共同 的 约定 ， 而 不 是 编程 语言 所 强制 的 。 如 果 程 
序 员 不 够 警惕 一 通常 是 因为 他 们 太 忙 ， 这 些 机 制 就 很 容易 被 忽视 。 

异常 处 理 将 错误 处 理 直 接 置 于 编程 语言 中 ， 有 时 甚至 置 于 操作 系统 中 。 异 常 是 一 种 对 象 
它 从 出 错 地 点 被 “ 抛 出 ”， 并 被 专门 设计 用 来 处 理 特定 类 型 错误 的 相应 的 异常 处 理 器 “捕获 ”。 
异常 处 理 就 像 是 与 程序 正常 执行 路 径 并 行 的 、 在 错误 发 生 时 执行 的 另 一 条 路 径 。 因 为 它 是 另 一 
条 完全 分 离 的 执行 路 径 ， 所 以 它 不 会 千 扰 正常 的 执行 代码 。 这 往往 使 得 代码 编写 变 得 简单 ， 因 
为 不 需要 被 迫 定期 检查 错误 。 此 外 ， 被 抛 出 的 异常 不 像 方法 返回 的 错误 值 和 方法 设置 的 用 来 表 
示 错误 系 件 的 标志 位 那样 可 以 被 忽略 。 异 常 不 能 被 忽略 ， 所 以 它 保证 一 定 会 在 某 处 得 到 处 理 。 
最 后 需要 指出 的 是 : 异常 提供 了 一 种 从 错误 状况 进行 可 靠 恢复 的 途径 。 现 在 不 再 是 只 能 退出 程 
序 ， 你 可 以 经 常 进行 校正 ， 并 恢复 程序 的 执行 ， 这 些 都 有 助 于 编写 出 更 健壮 的 程序 。 

Java 的 异常 处 理 在 众多 的 编程 语言 中 格外 引 人 注 目 ， 因 为 Java 一 开始 就 内 置 了 异常 处 理 ， 而 
且 强 制 你 必须 使 用 它 。 它 是 唯一 可 接受 的 错误 报告 方式 。 如 果 没 有 编写 正确 的 处 理 异常 的 代码 
那么 就 会 得 到 一 条 编译 时 的 出 错 消息 。 这 种 有 保障 的 一 致 性 有 了 时 会 使 得 错误 处 理 非常 容易 

值得 注意 的 是 ， 异 常 处 理 不 是 面向 对 象 的 特征 一 一 尽管 在 面向 对 象 语言 中 异常 常 被 表示 成 
为 一 个 对 象 。 异 常 处 理 在 面向 对 象 语言 出 现 之 前 就 已 经 存在 了 。 


1.12 并 发 编程 


在 计算 机 编程 中 有 一 个 基本 概念 ， 就 是 在 同一 时 刻 处 理 多 个 任务 的 思想 。 许 多 程序 设计 问 
题 都 要 求 ， 程 序 能 够 停 下 正在 做 的 工作 ， 转 而 处 理 某 个 其 他 问题 ， 然 后 再 返回 主 进程 。 有 许多 
方法 可 以 实现 这 个 目的 。 最 初 ， 程 序 员 们 用 所 掌握 的 有 关机 器 底层 的 知识 来 编写 中 断 服务 程序 ， 
主 进程 的 挂 起 是 通过 硬件 中 断 来 触发 的 。 尽 管 这 么 做 可 以 解决 问题 ， 但 是 其 难度 太 大 ， 而 且 不 
能 移植 ， 所 以 使 得 将 程序 移植 到 新 型 号 的 机 器 上 时 ， 既 费时 又 费力 。 

有 时 中 断 对 于 处 理 时 间 性 强 的 任务 是 必需 的 ， 但 是 对 于 大 量 的 其 他 问题 ， 我 们 只 是 想 把 问 
题 切 分 成 多 个 可 独立 运行 的 部 分 (任务 ) ， 从 而 提高 程序 的 响应 能 力 。 在 程序 中 ， 这 些 彼此 独立 
运行 的 部 分 称 之 为 线程 ， 上 述 概念 被 称 为 “并 发 "。 并 发 最 常见 的 例子 就 是 用 户 界面 。 通 过 使 用 
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任务 ， 用 户 可 以 在 揪 下 按钮 后 快速 得 到 一 个 响应 ， 而 不 用 被 迫 等 待 到 程序 完成 当前 任务 为 止 。 

通常 ， 线 程 只 是 一 种 为 单一 处 理 器 分 配 执行 时 间 的 手段 。 但 是 如 果 操 作 系统 支持 多 处 理 器 
那么 每 个 任务 都 可 以 被 指派 给 不 同 的 处 理 器 ， 并 且 它 们 是 在 真正 地 并 行 执行 。 在 语言 级 别 上 
多 线程 所 带 来 的 便利 之 一 便 是 程序 员 不 用 再 操心 机 器 上 是 有 多 个 处 理 器 还 是 只 有 一 个 处 理 器 
由 于 程序 在 逻辑 上 被 分 为 线程 ， 所 以 如 果 机 器 拥有 多 个 处 理 器 ， 那 么 程序 不 需要 特殊 调整 也 能 
执行 得 更 快 。 

所 有 这 些 都 使 得 并 发 看 起 来 相当 简单 ， 但 是 有 一 个 隐患 ， 共 享 资源 。 如 果 有 多 个 并 行 任务 
都 要 访问 同一 项 资源 ， 那 么 就 会 出 问题 。 例 如 ， 两 个 进程 不 能 同时 向 一 台 打印 机 发 送信 息 。 为 
了 解决 这 个 问题 ， 可 以 共享 的 资源 ， 例 如 打印 机 ， 必 须 在 使 用 期 间 被 锁定 。 因 此 ， 整 个 过 程 是 ; 
某 个 任务 锁定 某 项 资源 ， 完 成 其 任务 ， 然 后 释放 资源 锁 ， 使 其 他 任务 可 以 使 用 这 项 资源 。 

Java 的 并 发 是 内 置 于 语言 中 的 ，Java SE5 已 经 增添 了 大 量 额外 的 库 支持 。 


1.13 Java 与 Internet 


如 果 Java 仅 仅 只 是 众多 的 程序 设计 语言 中 的 一 种 ， 你 可 能 就 会 问 :为 什么 它 如 此 重要 ? 为 
什么 它 促使 计算 机 编程 语言 向 前 迈进 了 革命 性 的 一 步 ? 如 果 从 传统 的 程序 设计 观点 看 ， 问 题 的 
答案 似乎 不 太 明 显 。 尽 管 Java 对 于 解决 传统 的 单机 程序 设计 问题 非常 有 用 ， 但 同样 重要 的 是 ， 
它 解决 了 在 万 维 网 (WWW) 上 的 程序 设计 问题 。 

1.13.1 Web 是 什么 

Web 一 词 乍 一 看 有 点 神秘 ， 就 像 “ 网 上 冲浪 "、“ 表 现 "、“ 主 页 ”一 样 。 回 头 审视 它 的 真实 
面 角 有 助 于 对 它 的 理解 ， 但 是 要 这 么 做 就 必须 先 理解 客户 /服务 器 系统 ， 它 是 计算 技术 中 另 一 个 
充满 了 诸多 疑惑 的 话题 。 

1. 客户 /服务 器 计算 技术 

客户 /服务 器 系统 的 核心 思想 是 :系统 具有 一 个 中 央 信息 存储 池 (central repository of 
information) ， 用 来 存储 某 种 数据 ， 它 通常 存在 于 数据 库 中 ， 你 可 以 根据 需要 将 它 分 发 给 某 些 人 
员 或 机 器 集群 。 客 户 /服务 器 概念 的 关键 在 于 信息 存储 池 的 位 置 集中 于 中 央 ， 这 使 得 它 可 以 被 修 
改 ， 并 且 这 些 修改 将 被 传播 给 信息 消费 者 。 总 之 ， 信 息 存储 池 、 用 于 分 发 信息 的 软件 以 及 信息 
与 软件 所 驻 留 的 机 器 或 机 群 被 总 称 为 服务 器 。 驻 留 在 用 户 机 器 上 的 软件 与 服务 器 进行 通信 ， 以 
获取 信息 、 处 理 信息 ， 然 后 将 它们 显示 在 被 称 为 客户 机 的 用 户 机 器 上 。 

客户 /服务 器 计算 技术 的 基本 概念 并 不 复杂 。 问 题 在 于 你 只 有 单一 的 服务 器 ， 却 要 同时 为 多 
个 客户 服务 。 通 常 ， 这 会 涉及 数据 库 管理 系统 ， 因 此 设计 者 把 数据 “均衡 ”分 布 于 数据 表 中 ， 
以 取得 最 优 的 使 用 效果 。 此 外 ， 系 统 通常 允许 客户 在 服务 器 中 插入 新 的 信息 。 这 意味 着 必须 保 
证 一 个 客户 插入 的 新 数据 不 会 覆盖 另 一 个 客户 插入 的 新 数据 ， 也 不 会 在 将 其 添加 到 数据 库 的 过 
程 中 丢失 (这 被 称 为 事务 处 理 )。 如 果 客户 端 软件 发 生变 化 ， 那 么 它 必须 被 重新 编译 、 调 试 并 安 
装 到 客户 端 机 器 上 ， 事 实证 明 这 比 想像 的 要 更 加 复杂 与 费力 。 如 果 想 支持 多 种 不 同类 型 的 计算 
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机 和 操作 系统 ， 问 题 将 更 麻烦 。 最 后 还 有 一 个 最 重要 的 性 能 问题 ， 可 能 在 任意 时 刻 都 有 成 百 上 
千 的 客户 向 服务 器 发 出 请 求 ， 所 以 任何 小 的 延迟 都 会 产生 重大 影响 。 为 了 将 延迟 最 小 化 ， 程 序 
员 努 力 减 轻 处 理 任务 的 负载 ， 通 常 是 分 散 给 客户 端 机 器 处 理 ， 但 有 时 也 会 使 用 所 谓 的 中 间 件 将 
负载 分 散 给 在 服务 器 端的 其 他 机 器 。( 中 间 件 也 被 用 来 提高 可 维护 性 .) 

分 发 信息 这 个 简单 思想 的 复杂 性 实际 上 是 有 很 多 不 同 层次 的 ， 这 使 得 整个 问题 可 能 看 起 来 
高 深 莫 测 。 但 是 它 仍然 至 关 重 要 : 算 起 来 客户 /服务 器 计算 技术 大 概 占 了 所 有 程序 设计 行为 的 一 
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半 ， 从 制定 订单 、 信 用 卡 交易 到 包括 股票 市 场 、 科 学 计算 、 政 府 、 个 人 在 内 的 任意 类 型 的 数据 
分 发 。 过 去 我 们 所 做 的 ， 都 是 针对 某 个 问题 发 明 一 个 单独 的 解决 方案 ， 所 以 每 一 次 都 要 发 明 一 
个 新 的 方案 。 这 些 方案 难以 开发 且 难 以 使 用 ， 而 且 用 户 对 每 一 个 方案 都 要 学 习 新 的 接口 。 因 此 ， 
整个 客户 /服务 器 问题 需要 彻底 解决 。 

2. Web 就 是 一 台 巨型 服务 器 

Web 实 际 上 就 是 一 个 巨型 客户 /服务 器 系统 ， 但 稍微 差 一 点 ， 因 为 所 有 的 服务 器 和 客户 机 都 
同时 共存 于 同一 个 网 络 中 。 你 不 需要 了 解 这 些 ， 因 为 你 所 要 关心 的 只 是 在 某 一 时 刻 怎样 连接 到 
一 台 服 务 器 上 ， 并 与 之 进行 交互 〈 即 便 你 可 能 要 满 世 界 地 查找 你 想 要 的 服务 器 ) 。 

最 初 只 有 一 种 很 简单 的 单 向 过 程 : 你 对 某 个 服务 器 产生 一 个 请 求 ， 然 后 它 返 回 给 你 一 个 文 
件 ， 你 的 机 器 (也 就 是 客户 机 ) 上 的 浏览 器 软件 根据 本 地 机 器 的 格式 来 解读 这 个 文件 。 但 是 很 
快 人 们 就 希望 能 够 做 得 更 多 ， 而 不 仅仅 是 从 服务 器 传递 回 页 面 。 人 们 希望 实现 完整 的 客户 /服务 
器 能 力 ， 使 得 客户 可 以 将 信息 反馈 给 服务 器 。 例 如 ， 在 服务 器 上 进行 数据 库 查找 ， 将 新 信息 添 
加 到 服务 器 以 及 下 订单 (这 需要 特殊 的 安全 措施 ) 。 这 些 变革 ， 正 是 我 们 在 Web 发 展 过 程 中 所 目 
有 睹 的 。 

Web 浏 览 器 向 前 跨 进 了 一 大 步 ， 它 包含 了 这 样 的 概念 : 一段 信息 不 经 修改 就 可 以 在 任意 型 
号 的 计算 机 上 显示 。 然 而 ， 最 初 的 浏览 器 仍然 相当 原始 ， 很 快 就 因为 加 诸 于 其 上 的 种 种 需要 而 
陷 和 人 困境。 浏览 器 并 不 具备 显著 的 交互 性 ， 而 且 它 趋向 于 使 服务 器 和 Internet 阻 塞 ， 因 为 在 任何 
有 时候， 只 要 你 需要 完成 通过 编程 来 实现 的 任务 ， 就 必须 将 信息 发 回 到 服务 器 去 处 理 。 这 使 得 即 
便 是 发 现 你 的 请 求 中 的 拼写 错误 也 要 花 去 数秒 甚至 是 数 分 钟 的 时 间 。 因 为 浏览 器 只 是 一 个 观察 
器 ， 因 此 它 甚至 不 能 执行 最 简单 的 计算 任务 。( 另 一 方面 ， 它 却 是 安全 的 ， 因 为 它 在 你 的 本 地 机 
器 上 不 会 执行 任何 程序 ， 而 这 些 程序 有 可 能 包含 bug 和 病毒 .) 

为 了 解决 这 个 问题 ， 人 们 采用 了 各 种 不 同 的 方法 。 首 先 ， 图 形 标准 得 到 了 增强 ， 使 得 在 浏 
览 器 中 可 以 播放 质量 更 好 的 动画 和 视频 。 剩 下 的 问题 通过 引入 在 客户 端 浏览 器 中 运行 程序 的 能 
力 就 可 以 解决 。 这 被 称 为 “客户 端 编程 ”"。 

1.13.2 客户 端 编程 

Web 最 初 的 “服务 器 一 浏览 器 ”设计 是 为 了 能 够 提供 交互 性 的 内 容 ， 但 是 其 交互 性 完全 由 服 
务 器 提供 。 服 务 器 产生 静态 页 面 ， 提 供给 只 能 解释 并 显示 它们 的 客户 端 浏览 器 。 基 本 的 HTML 
(HyperText Markup Language， 超 文本 标记 语言 ) 包含 有 简单 的 数据 收集 机 制 : 文本 输入 框 、 复 
选 框 、 单 选 框 、 列 表 和 下 拉 式 列表 以 及 按钮 一 - 它 只 能 被 编程 来 实现 复位 表单 上 的 数据 或 提交 
表单 上 的 数据 给 服务 器 。 这 种 提交 动作 通过 所 有 的 Web 服 务 器 都 提供 的 通用 网关 接口 (common 
gateway interface, CGI) 传递 。 提 交 内 容 会 告诉 CGI 应 该 如 何 处 理 它 。 最 常见 的 动作 就 是 运行 一 
个 在 服务 器 中 常 被 命名 为 “cgi-bin” 的 目录 下 的 一 个 程序 。( 当 点 击 了 网 页 上 的 按钮 时 ， 如 果 观 
察 浏览 器 窗口 顶部 的 地 址 ， 有 时 可 以 看 见 “cgi-bin” 的 字样 混迹 在 一 串 宛 长 和 不 知 所 云 的 字符 
中 。) 几乎 所 有 的 语言 都 可 以 用 来 编写 这 些 程序 ，Perl 已 经 成 为 最 常见 的 选择 ， 因 为 它 被 设计 用 
来 处 理 文本 ， 并 且 是 解释 型 语言 ， 因 此 无 论 服 务 器 的 处 理 器 和 操作 系统 如 何 ， 它 都 适 于 安装 。 
然而 ，Python (www.Python.org) 已 对 其 产生 了 重大 的 冲击 ， 因 为 它 更 强大 且 更 简单 。 

当今 许多 有 影响 力 的 网 站 完全 构建 于 CGI 之 上 的 ， 实 际 上 你 几乎 可 以 通过 CGI 做 任何 事 。 然 
而 ， 构 建 于 CGI 程序 之 上 的 网 站 可 能 会 迅速 变 得 过 于 复杂 而 难以 维护 ， 并 同时 产生 响应 时 间 过 长 
的 问题 。CGI 程 序 的 响应 时 间 依赖 于 所 必须 发 送 的 数据 量 的 大 小 ， 以 及 服务 器 和 Intermnet 的 负载 。 
(此 外 ， 启 动 CGI 程 序 也 相当 慢 .) Web 的 最 初 设计 者 们 并 没有 预见 到 网 络 带 宽 被 人 们 开发 的 各 种 
应 用 迅速 耗 尽 。 例 如 ， 任 何 形式 的 动态 图 形 处 理 几 乎 都 不 可 能 连贯 地 执行 ， 因 为 图 形 交互 格式 
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(graphic interchange format, GIF) 的 文件 必须 在 服务 器 端 创建 每 一 个 图 形 版 本 ， 并 发 送 给 客户 
端 。 再 比如 ， 你 肯定 经 历 过 对 Web 输 入 表单 进行 数据 验证 的 过 程 : 你 按 下 网 页 上 的 提交 按钮 ， 
数据 被 发 送 回 服务 器 ;服务 器 启动 一 个 CGI 程序 来 检查 、 发 现 错误 ， 并 将 错误 组 装 为 一 个 HTML 
页 面 ， 然 后 将 这 个 页 面 发 回 给 你 ， 之 后 你 必须 回 退 一 个 页 面 ， 然 后 重新 再 试 。 这 个 过 程 不 仅 很 
慢 ， 而 且 不 太 优 雅 。 

问题 的 解决 方法 就 是 客户 端 编程 。 大 多 数 运行 Web 浏 览 器 的 机 器 都 是 能 够 执行 大 型 任务 的 
强 有 力 的 引擎 。 在 使 用 原始 的 静态 HTML 方 式 的 情况 下 ， 它 们 只 是 闲 在 那里 ， 等 着 服务 器 送 来 
下 一 个 页 面 。 客 户 端 编程 意味 着 Web 浏 览 器 能 用 来 执行 任何 它 可 以 完成 的 工作 ， 使 得 返回 给 用 
户 的 结果 更 加 迅捷 ， 而 且 使 得 你 的 网 站 更 加 具有 交互 性 。 . 

客户 端 编程 的 问题 是 ， 它 与 通常 意义 上 的 编程 十 分 不 同 ， 参 数 几乎 相同 ， 而 平台 却 不 同 。 
Web 浏 览 器 就 像 一 个 功能 受 限 的 操作 系统 。 最 终 ， 你 仍然 必须 编写 程序 ， 而 且 还 得 处 理 那 些 令 
人 头 晤 眼花 的 成 堆 的 问题 ， 并 以 客户 端 编程 的 方式 来 产生 解决 方案 。 本 节 剩 下 的 部 分 对 客户 端 
编程 的 问题 和 方法 作 一 概述 。 

1. 插件 

客户 端 编程 所 迈 出 的 最 重要 的 一 步 就 是 插件 (plug-in) 的 开发 。 通 过 这 种 方式 ， 程 序 员 可 
以 下 载 一 段 代码 ， 并 将 其 插入 到 浏览 器 中 适当 的 位 置 ， 以 此 来 为 浏览 器 添加 新 功能 。 它 告诉 浏 
览 器 :从 现在 开始 ， 你 可 以 采取 这 个 新 行动 了 (只 需要 下 载 一 次 插件 即 可 )。 某 些 更 快 更 强大 的 
行为 都 是 通过 插件 添加 到 服务 器 中 的 。 但 是 编写 插件 并 不 是 件 轻松 的 事 ， 也 不 是 构建 某 特定 网 
站 的 过 程 中 所 要 做 的 事情 。 插 件 对 于 客户 端 编程 的 价值 在 于 ， 它 允许 专家 级 的 程序 员 不 需 经 过 
浏览 器 生产 厂商 的 许可 ， 就 可 以 开发 某 种 语言 扩展 ， 并 将 它们 添加 到 服务 器 中 。 因 此 ， 插 件 提 
供 了 一 个 “后 门 "， 使 得 可 以 创建 新 的 客户 端 编程 语言 (但 是 并 不 是 所 有 的 客户 端 编程 语言 都 是 
以 插件 的 形式 实现 的 )。 

2. 脚本 语言 

插件 引发 了 浏览 器 脚本 语言 (scripting language) 的 开发 。 通 过 使 用 某 种 脚本 语言 ， 你 可 
以 将 客户 端 程序 的 源 代 码 直接 供 入 到 HTML 页 面 中 ， 解 释 这 种 语言 的 插件 在 HTML 页 面 被 显示 
时 自动 激活 。 脚 本 语言 先天 就 相当 易于 理解 ， 因 为 它们 只 是 作为 HTML 页 面 一 部 分 的 简单 文本 ， 
当 服务 器 收 到 要 获取 该 页 面 的 请 求 时 ， 它 们 可 以 被 快速 加 载 。 此 方法 的 缺点 是 代码 会 暴露 给 任 
何人 去 浏览 (或 窃取 )。 但 是 ， 通 常 不 会 使 用 脚本 语言 去 做 相当 复杂 的 事情 ， 所 以 这 个 缺点 并 
不 太 严重 。 

如 果 你 期 望 有 一 种 脚本 语言 在 Web 浏 览 器 不 需要 任何 插件 的 情况 下 就 可 以 得 到 支持 ， 那 它 
非 JavaScript 莫 属 ( 它 与 Java 之 间 只 存在 表面 上 的 相似 ， 要 想 使 用 它 ， 你 必须 在 额外 的 学 习 曲 线 
上 攀 爬 。 它 之 所 以 这 样 被 命名 只 是 因为 想 赶 上 Java 潮 流 )。 遗 憾 的 是 ， 大 多 数 Web 浏 览 器 最 初 都 
是 以 彼此 相 异 的 方式 来 实现 对 JavaScript 的 支持 的 ， 这 种 差异 甚至 存在 于 同一 种 浏览 器 的 不 同 版 
本 之 间 。 以 ECMAScript 的 形式 实现 的 JavaScript 的 标准 化 有 助 于 此 问题 的 解决 ， 但 是 不 同 的 浏览 
器 为 了 跟 上 这 一 标准 化 趋势 已 经 花费 了 相当 长 的 时 间 (并 且 这 种 努力 由 于 微软 一 直 在 推进 它 自 
己 的 VBScript 形 式 的 标准 化 日 程 而 显得 无 所 帮助 ，VBScript 与 JavaScript 之 间 也 存在 着 暧昧 的 相似 
性 )。 通 常 ， 你 必须 以 JavaScript 的 某 种 最 小 公分 母 形式 来 编程 ， 以 使 得 你 的 程序 可 以 在 所 有 的 浏 
览 器 上 运行 。JavaScript 的 错误 处 理 的 调试 只 能 一 团 精 来 形容 。 作 为 其 使 用 艰难 的 证 据 ， 我 们 可 
以 看 到 直到 最 近 才 有 人 创建 了 真正 复杂 的 JavaScript 脚 本 片段 (Google 在 GMail) ， 并 且 编写 这 样 
的 脚本 需要 超然 的 奉献 精神 和 超 高 的 专业 技巧 。 

这 也 表明 ， 在 Web 浏 览 器 内 部 使 用 的 脚本 语言 实际 上 总 是 被 用 来 解决 特定 类 型 的 问题 ， 主 
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要 是 用 来 创建 更 丰富 、 更 具有 交互 性 的 图 形 化 用 户 界面 (graphic user interface, GUI)。 但 是 ， 脚 
本 语言 可 以 解决 客户 端 编程 中 所 遇 到 的 百 分 之 八 十 的 问题 。 你 的 问题 可 能 正好 落 在 这 百 分 之 八 
十 的 范围 之 内 ， 由 于 脚本 语言 提供 了 更 容易 、 更 快捷 的 开发 方式 ， 因 此 你 应 该 在 考虑 诸如 Java 
这 样 的 更 复杂 的 解决 方案 之 前 ， 先 考虑 脚本 语言 。 

3. Java 

如 果 脚 本 语言 可 以 解决 客户 端 编程 百 分 之 八 十 的 问题 的 话 ， 那 么 剩 下 那 百 分 之 二 十 ( 那 才 
是 真正 难 哨 的 硬骨头 ) 又 该 怎么 办 呢 ? Java 是 处 理 它们 最 流行 的 解决 方案 。Java 不 仅 是 一 种 功能 
强大 的 、 安 全 的 、 跨 平台 的 、 国 际 化 的 编程 语言 ， 而 且 它 还 在 不 断 地 被 扩展 ， 以 提供 更 多 的 语 
言 功 能 和 类 库 ， 能 够 优雅 地 处 理 在 传统 编程 语言 中 很 难 解决 的 问题 ， 例 如 并 发 、 数 据 库 访问 、 
网 络 编程 和 分 布 式 计算 。Java 是 通过 applet 以 及 使 用 Java Web Start 来 进行 客户 端 编程 的 。 

applet 是 只 在 Web 浏 览 器 中 运行 的 小 程序 ， 它 是 作为 网 页 的 一 部 分 而 自动 下 载 的 (就 像 网 页 
中 的 图 片 被 自动 下 载 一 样 )。 当 applet 被 激活 时 ， 它 便 开始 执行 一 个 程序 ， 这 正 是 它 优雅 之 处 
它 提供 一 种 分 发 软件 的 方法 ， 一 旦 用 户 需要 客户 端 软件 时 ， 就 自动 从 服务 器 把 客户 端 软件 分 发 
给 用 户 。 用 户 获取 最 新 版 本 的 客户 端 软件 时 不 会 产生 错误 ， 而 且 也 不 需要 很 麻烦 的 重新 安装 过 
程 。Java 的 这 种 设计 方式 ， 使 得 程序 员 只 需 创建 单一 的 程序 ， 而 只 要 一 台 计算 机 有 浏览 器 ， 且 
浏览 器 具有 内 置 的 Java 解 释 器 〈 大 多 数 的 机 器 都 如 此 ) ， 那 么 这 个 程序 就 可 以 自动 在 这 台 计 算 机 
上 运行 。 由 于 Java 是 一 种 成 熟 的 编程 语言 ， 所 以 在 提出 对 服务 器 的 请 求 之 前 和 之 后 ， 可 以 在 客 
户 端 尽 可 能 多 地 做 些 事情 。 例 如 ， 不 必 跨 网 络 地 发 送 一 张 请 求 表单 来 检查 自己 是 否 填写 了 错误 
的 日 期 或 其 他 参数 ， 客 户 端 计算 机 就 可 以 快速 地 标 出 错误 数据 ， 而 不 用 等 待 服务 器 作出 标记 并 
给 你 传 回 图 片 。 这 不 仅 立即 就 获得 了 高 速度 和 快速 的 响应 能 力 ， 而 且 也 降低 了 网 络 流量 和 服务 
器 负载 ， 从 而 不 会 使 整个 Intemet 的 速度 都 慢 了 下 来 。 

4. 备 选 方案 

老实 说 ，Java applet 没 有 达到 当初 它 所 吹 吐 的 境界 。 当 Java 首 度 出 现时 ， 似 乎 大 家 最 欢欣 鼓 
舞 的 莫 过 于 applet 了 ， 因 为 它们 最 终 将 解决 严峻 的 客户 端 可 编程 性 问题 ， 从 而 提高 基于 互联 网 的 
应 用 的 可 响应 性 ， 同 时 降低 它们 对 带宽 的 需求 。 人 们 展望 到 了 大 量 的 可 能 性 。 

实际 上 ， 你 可 以 发 现在 Web 上 确实 存在 一 些 非常 灵巧 的 applet， 但 是 压倒 性 的 向 applet 的 迁移 
却 始终 未 发 生 。 这 其 中 最 大 的 问题 可 能 在 于 安装 Java 运 行 时 环境 (IRE) 所 必需 的 10MB 带 宽 对 
于 一 般 的 用 户 来 说 过 于 恐怖 了 ， 而 微软 没有 选择 在 IE (Intemet Explorer) 中 包含 IRE 这 一 事实 也 
许 就 此 已 经 封杀 了 applet 的 命运 。 无 论 怎样 ，Java applet 始 终 没有 得 到 大 规模 应 用 。 

尽管 如 此 ，applet 和 Java Web Start 应 用 在 某 些 情况 下 仍旧 很 有 价值 。 无 论 何 时 ， 只 要 你 想 控 
制 用 户 的 机 器 ， 例 如 在 一 个 公司 的 内 部 ， 使 用 这 些 技术 来 发 布 和 更 新 客户 端 应 用 就 显得 非常 恰 
当 ， 并 且 这 可 以 节省 大 量 的 时 间 、 人 力 和 财力 ， 特 别 是 你 需要 频繁 地 更 新 的 时 候 。 

在 “图 形 化 用 户 界面 ”一 章 中 ， 我 们 将 看 到 一 种 折 中 的 新 技术 ，Macromedia 的 Flex， 它 多 
许 你 创建 基于 Flash 的 与 applet 相 当 的 应 用 。 因 为 Flash Player 在 超过 98% 的 Web 浏 览 器 上 都 可 用 
(包含 Windows, Linux 和 Mac 操 作 系统 上 的 浏览 器 ) ， 因 此 它 被 认为 是 事实 上 已 被 接受 的 标准 。 安 
装 和 更 新 Flash Player 都 十 分 快捷 。ActionScript 语 言 是 基于 ECMAScript 的 ， 因 此 我 们 对 它 应 该 很 
熟悉 ， 但 是 Flex 使 得 我 们 在 编程 时 无 需 担 心 浏览 器 相关 性 ， 因 此 ， 它 远 比 JavaScript 要 吸引 人 得 
多 。 对 于 客户 端 编程 而 言 ， 这 是 一 种 值得 考虑 的 备 选 方案 。 

5. NET 和 C# 

曾几何时 ，Java applet 的 主要 竞争 对 手 是 微软 的 ActiveX 一 一 尽管 它 要 求 客户 端 必须 运行 
Windows 平 台 。 从 那 以 后 ， 微 软 以 .NET 平 台 和 C# 编 程 语言 的 形式 推出 了 与 Java 全 面 竞争 的 对 
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手 。.NET 平 台大 致 相当 于 Java 虚 拟 机 (JVM， 即 执行 Java 程 序 的 软件 平台 ) 和 Java 类 库 ， 而 C# 毫 
无 疑问 与 Java 有 类 似 之 处 。 这 当然 是 微软 在 编程 语言 与 编程 环境 这 块 竞技 场 上 所 做 出 的 最 出 色 
的 成 果 。 当 然 ， 他 们 有 相当 大 的 有 利 条 件 : 他 们 可 以 看 得 到 Java 在 什么 方面 做 得 好 ， 在 什么 方 
面 做 得 还 不 够 好 ， 然 后 基于 此 去 构建 ， 并 要 具备 Java 不 具备 的 优点 。 这 是 自从 Java 出 现 以 来 ， 
Java 所 磁 到 的 真正 的 竞争 。 因 此 ，Sun 的 Java 设 计 者 们 已 经 认真 仔细 地 去 研究 了 C#， 以 及 为 什么 
程序 员 们 可 能 会 转 而 使 用 它 ， 然 后 通过 在 Java SES 中 对 Java 做 出 的 重大 改进 而 做 出 了 回应 。 

目前 ，.NET 主 要 受 攻击 的 地 方 和 人 们 所 关心 的 最 重要 的 问题 就 是 ， 微 软 是 否 会 允许 将 它 完 
全 地 移植 到 其 他 平台 上 。 微 软 宣称 这 人 么 做 没有 问题 ， 而 且 Mono 项 目 《www.go-mono.com) 已 经 
有 了 一 个 在 Linux 上 运行 的 ,NET 的 部 分 实现 ， 但 是 ， 在 该 实现 完成 及 微软 不 会 排斥 其 中 的 任何 部 
分 之 前 ，NET 作 为 一 种 跨 平台 的 解决 方案 仍旧 是 一 场 高 风险 的 赌博 。 

6. Internet 与 Intranet 

Web 是 最 常用 的 解决 客户 /服务 器 问题 的 方案 ， 因 此 ， 即 便 是 解决 这 个 问题 的 一 个 子 集 ， 特 
别 是 公司 内 部 的 典型 的 客户 /服务 器 问题 ， 也 一 样 可 以 使 用 这 项 技术 。 如 果 采 用 传统 的 客户 /服务 
器 方式 ， 可 能 会 遇 到 客户 端 计算 机 有 多 种 型 号 的 问题 ， 也 可 能 会 遇 到 安装 新 的 客户 端 软 件 的 麻 
烦 ， 而 它们 都 可 以 很 方便 地 通过 Web 浏 览 器 和 客户 端 编程 得 以 解决 。 当 Web 技 术 仅 限 用 于 特定 公 
司 的 信息 网 络 时 ， 它 就 被 称 为 Intranet (企业 内 部 网 ) 。Intranet 比 Internet 提 供 了 更 高 的 安全 性 ， 
因为 可 以 物理 地 控制 对 公司 内 部 服务 器 的 访问 。 从 培训 的 角度 看 ， 似 乎 一 旦 人 们 理解 了 浏览 器 
的 基本 概念 后 ， 对 他 们 来 说 ， 处 理 网 页 和 applet 的 外 观 差异 就 会 容易 得 多 ， 因 此 对 新 型 系统 的 学 
习 曲 线 也 就 减缓 了 。 

安全 问题 把 我 们 带 到 了 一 个 领域 ， 这 似乎 是 在 客户 端 编程 世界 自动 形成 的 。 如 果 程 序 运行 
在 Internet 之 上 ， 那 么 就 不 可 能 知道 它 将 运行 在 什么 样 的 平台 之 上 ， 因 此 ， 要 格外 小 心 ， 不 要 传 
播 有 bug 的 代码 。 你 需要 跨 平台 的 、 安 全 的 语言 ， 就 像 脚 本 语言 和 Java。 


如 果 程序 运行 与 Intranet 上 ， 那 么 可 能 会 受到 不 同 的 限制 。 企 业内 所 有 的 机 器 都 采用 


Intel/Windows 平 台 并 不 是 什么 稀奇 的 事 。 在 Intranet 上 ， 你 可 以 对 你 自己 的 代码 质量 负责 ， 并且 
在 发 现 bug 之 后 可 以 修复 它们 。 此 外 ， 你 可 能 已 经 有 了 以 前 使 用 更 传统 的 客户 /服务 器 方式 编写 
的 遗留 代码 ， 因 此 ， 你 必须 在 每 一 次 升级 时 都 要 物理 地 重 装 客户 端 程序 。 在 安装 升级 程序 时 所 
浪费 的 时 间 是 迁移 到 浏览 器 方式 上 的 最 主要 的 原因 ， 因 为 在 浏览 器 方式 下 ， 升 级 是 透明 的 、 自 
动 的 〈Java Web Start 也 是 解决 此 问题 的 方式 之 一 ) 。 如 果 你 身 处 这 样 的 Intranet 之 中 ， 那么 最 有 意 
义 的 方式 就 是 选择 一 条 能 够 使 用 现 有 代码 库 的 最 短 的 捷径 ， 而 不 是 用 一 种 新 语言 重新 编写 你 的 
代码 。 

当面 对 各 种 令 人 眼花 统 乱 的 解决 客户 端 编程 问题 的 方案 时 ， 最 好 的 方法 就 是 进行 性 价 比分 
析 。 认 真 考虑 问题 的 各 种 限制 ， 然 后 思考 哪 种 解决 方案 可 以 成 为 最 短 的 捷径 。 既 然 客户 端 编程 
仍然 需要 编程 ， 那 么 针对 自己 的 特殊 应 用 选取 最 快 的 开发 方式 总 是 最 好 的 做 法 。 为 那些 在 程序 
开发 中 不 可 避免 的 问题 提早 作 准 备 是 一 种 积极 的 态度 。 
1.13.3 服务 器 端 编程 

前 面 的 讨论 忽略 了 服务 器 端 编程 的 话题 ， 它 是 Java 已 经 取得 巨大 成 功 的 因素 之 一 。 当 提出 
对 服务 器 的 请 求 后 ， 会 发 生 什 么 呢 ? 大 部 分 时 间 ， 请 求 只 是 要 求 “ 给 我 发 送 一 个 文件 "， 之 后 浏 
览 器 会 以 某 种 适当 的 形式 解释 这 个 文件 ， 例 如 将 其 作为 HTML 页面 、 图 片 、Java applet 或 脚本 程 
序 等 来 解释 。 

更 复杂 的 对 服务 器 的 请 求 通常 涉及 数据 库 事务 。 常 见 的 情形 是 复杂 的 数据 库 搜索 请 求 ， 然 
后 服务 器 将 结果 进行 格式 编排 ， 使 其 成 为 一 个 HTML 页 面 发 回 给 客户 端 。( 当 然 ， 如 果 客 户 端 通 
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过 Java 或 脚本 程序 具备 了 更 多 的 智能 ， 那 么 服务 器 可 以 将 原始 的 数据 发 回 ， 然 后 在 客户 端 进行 
格式 编排 ， 这 样 会 更 快 ， 而 且 服务 器 的 负载 将 更 小 .) 另 一 种 常见 情形 是 ， 当 你 要 加 入 一 个 团体 
或 下 订单 时 ， 可 能 想 在 数据 库 中 注册 自己 的 名 字 ， 这 将 涉及 对 数据 库 的 修改 。 这 些 数据 库 请 求 
必须 通过 服务 器 端的 某 些 代码 来 处 理 ， 这 就 是 所 谓 的 服务 器 端 编 程 。 过 去 ， 服 务 器 端 编程 都 是 
通过 使 用 Perl、Python、C++ 或 其 他 某 种 语言 编写 CGI 程 序 而 实现 的 ， 但 却 造成 了 从 此 之 后 更 加 
复杂 的 系统 。 其 中 就 包括 基于 Java 的 Web 服 务 器 ， 它 让 你 用 Java 编 写 被 称 为 serviet 的 程序 来 实现 
服务 器 端 编程 。servlet 及 其 衍生 物 JSP， 是 许多 开发 网 站 的 公司 迁移 到 Java 上 的 两 个 主要 的 原因 
尤其 是 因为 它们 消除 了 处 理 具 有 不 同 能 力 的 浏览 器 时 所 遇 到 的 问题 。 服 务 器 端 编 程 的 话题 在 
《企业 Java 编 程 思 想 》(Thinking in Enterprise Java) 一 书 中 有 所 论述 。 


1.14 总 结 


你 知道 过 程 型 语言 看 起 来 像 什么 样子 ,数据 定义 和 函数 调用 。 想 了 解 此 类 程序 的 含义 ， 你 
必须 忙 上 一 阵 ， 需 要 通读 函数 调用 和 低层 概念 ， 以 在 脑海 里 建立 一 个 模型 。 这 正 是 我 们 在 设计 
过 程式 程序 时 ， 需 要 中 间 表示 形式 的 原因 。 这 些 程序 总 是 容易 把 人 搞 糊 涂 ， 因 为 它们 使 用 的 表 
示 术 语 更 加 面向 计算 机 而 不 是 你 要 解决 的 问题 。 

因为 OOP 在 你 能 够 在 过 程 型 语言 中 找到 的 概念 的 基础 上 ， 又 添加 了 许多 新 概念 ， 所 以 你 可 
能 会 很 自然 地 假设 : 由 此 而 产生 的 Java 程 序 比 等 价 的 过 程 型 程序 要 复杂 得 多 。 但 是 ， 你 会 感到 
很 惊喜 : 编写 良好 的 Java 程 序 通常 比 过 程 型 程序 要 简单 得 多 ， 而 且 也 易于 理解 得 多 。 你 看 到 的 
只 是 有 关 下 面 两 部 分 内 容 的 定义 ; 用 来 表示 问题 空间 概念 的 对 象 (而 不 是 有 关 计 算 机 表示 方式 
的 相关 内 容 )， 以 及 发 送 给 这 些 对 象 的 用 来 表示 在 此 空间 内 的 行为 的 消息 。 面 向 对 象 程序 设计 带 
给 人 们 的 喜悦 之 一 就 是 : 对 于 设计 良好 的 程序 ， 通 过 阅读 它 就 可 以 很 容易 地 理解 其 代码 。 通 常 ， 
其 代码 也 会 少 很 多 ， 因 为 许多 问题 都 可 以 通过 重用 现 有 的 类 库 代码 而 得 到 解决 。 

OOP 和 Java 也 许 并 不 适合 所 有 的 人 。 重 要 的 是 要 正确 评估 自己 的 需求 ， 并 决定 Java 是 否 能 够 
最 好 地 满足 这 些 需求 ， 还 是 使 用 其 他 编程 系统 (包括 你 当前 正在 使 用 的 ) 才 是 更 好 的 选择 。 如 
果 知 道 自己 的 需求 在 可 预见 的 未 来 会 变 得 非常 特殊 化 ， 并 且 Java 可 能 不 能 满足 你 的 具体 限制 ， 
那么 就 应 该 去 考察 其 他 的 选择 (我 特别 推荐 读者 看 看 Python， 参 见 www.Python.org) 。 即 使 最 终 
仍旧 选择 Java 作 为 编程 语言 ， 至 少 也 要 理解 还 有 哪些 选项 可 供 选择 ， 并 且 对 为 什么 选择 这 个 方 


向 要 有 清楚 的 认识 。 





第 2 章 一 切 都 是 对 象 


“如 果 我 们 说 另 一 种 不 同 的 语言 ， 那 么 我 们 就 会 发 觉 一 个 有 些 不 同 的 世界 .” 
—Luduing Wittgerstein (1889-1951) 
尽管 Java 是 基于 C++ 的 ， 但 是 相 比 之 下 ，Java 是 一 种 更 “纯粹 ”的 面向 对 象 程序 设计 语言 。 
C++ 和 Java 都 是 混合 / 杂 合 型 语言 。 但 是 ，Java 的 设计 者 认为 这 种 杂 合 性 并 不 像 在 C++ 中 那么 
重要 。 杂 合 型 语言 允许 多 种 编程 风格 ，C++ 之 所 以 成 为 一 种 杂 合 型 语言 主要 是 因为 它 支持 与 C 语 
言 的 向 后 兼容 。 因 为 Ct+ 是 C 的 一 个 超 集 ， 所 以 势必 包括 许多 C 语 言 不 具备 的 特性 ， 这 些 特性 使 
C++ 在 某 些 方面 显得 过 于 复杂 。 

Java 语 言 假设 我 们 只 进行 面向 对 象 的 程序 设计 。 也 就 是 说 ， 在 开始 用 Java 进 行 设 计 之 前 ， 必 
须 将 思想 转换 到 面向 对 象 的 世界 中 来 。 这 个 入 门 基本 功 ， 可 以 使 你 具备 使 用 这 样 一 种 编程 语言 
编程 的 能 力 ， 这 种 语言 学 习 起 来 更 简单 ， 也 比 许多 其 他 OOP 语 言 更 易 用 。 在 本 章 ， 我 们 将 看 到 
Java 程 序 的 基本 组 成 部 分 ， 并 体会 到 在 Java 中 (几乎) 一 切 都 是 对 象 。 


2.1 用 引用 操纵 对 象 


每 种 编程 语言 都 有 自己 的 操纵 内 存 中 元 素 的 方式 。 有 时 候 ， 程 序 员 必 须 注意 将 要 处 理 的 数 
据 是 什么 类 型 。 你 是 直接 操纵 元 素 ， 还 是 用 某 种 基于 特殊 语法 的 间接 表示 〈 例 如 C 和 C++ 里 的 指 
针 ) 来 操纵 对 象 ? 

所 有 这 一 切 在 Java 里 都 得 到 了 简化 。 一 切 都 被 视 为 对 象 ， 因 此 可 采用 单一 固定 的 语法 。 尽 
管 一 切 都 看 作对 象 ， 但 操纵 的 标识 符 实际 上 是 对 象 的 一 个 “引用 ” (reference) 。 。 可 以 将 这 一 
情形 想像 成 用 遥控 器 (引用 ) 来 操纵 电视 机 (对象 )。 只 要 所 住 这 个 遥控 器 ， 就 能 保持 与 电视 机 
的 连接 。 当 有 人 想 改变 频道 或 者 碱 小 音量 时 ， 实 际 操控 的 是 允 控 器 (引用 ) ， 再 由 遥控 器 来 调控 
电视 机 (对象)。 如 果 想 在 房间 里 四 处 走 走 ， 同 时 仍 能 调控 电视 机 ， 那 么 只 需 携带 走 控 器 (引用 ) 
而 不 是 电视 机 (对象)。 

此 外 ， 即 使 没有 电视 机 ， 蜗 控 器 亦 可 独立 存在 。 也 就 是 说 ， 你 拥有 一 个 引用 ， 并 不 一 定 需 
要 有 一 个 对 象 与 它 关联 。 因 此 ， 如 果 想 操纵 一 个 词 或 句子 ， 则 可 以 创建 一 个 String 引 用 : 


String si 


但 这 里 所 创建 的 只 是 引用 ， 并 不 是 对 象 。 如 果 此 时 向 s 发 送 一 个 消息 ， 就 会 返回 一 个 运行 时 


O 这 可 能 会 引起 争论 。 有 人 认为 :“ 很 明显 ， 它 是 一 个 指针 .” 但 是 这 种 说 法 是 基于 底层 实现 的 某 种 假设 。 并 且 ， 
Java 中 的 引用 ， 在 语法 上 更 接近 C++ 的 引用 而 不 是 指针 。 本 书 的 第 1 版 中 ， 我 选择 发 明 一 个 新 术语 “句柄 
(handle)” 来 表示 这 一 概念 ， 因 为 ，Java 的 引用 和 C++ 的 引用 毕 竞 存在 一 些 重 大 差异 。 我 当时 正在 脱离 C++ 阵 
营 ， 而 且 也 不 想 使 那些 已 经 习惯 C++ 语言 的 程序 员 (我 想 他 们 将 来 会 是 最 大 的 、 热 圳 于 Java 的 群体 ) 感到 迷惑 。 
在 第 2 版 中 ， 我 决定 换 回 这 个 最 为 广泛 使 用 的 术语 -一 “引用 "。 并 且 ， 那 些 从 C++ 阵营 转换 过 来 的 人 们 ， 理 应 
更 会 处 理 引 用 ， 而 不 是 仅仅 理解 “引用 ”这 个 术语 ， 因 而 他 们 也 会 全 心 全 意 投入 其 中 的 。 尽 管 如 此 ， 还 是 有 
人 不 同意 用 “引用 ”这 个 术语 。 我 曾经 读 到 的 一 本 书 这 样 说 :“Java 所 支持 的 “ 按 址 传递 ”是 完全 错误 的 "， 因 
为 Java 对 象 标识 符 ( 按 那 位 作者 所 说 ) 实际 上 是 “对 象 引用 ”"。 并 且 他 接着 说 任何 事物 都 是 “ 按 值 传递 ”的 。 
也 许 有 人 会 赞成 这 种 精确 却 让 人 费解 的 解释 ， 但 我 认为 我 的 这 种 方法 可 以 简化 概念 上 的 理解 并 且 不 会 伤害 到 
任何 事物 。( 好 了 ， 那 些 语言 专家 可 能 会 说 我 在 后 说 ， 但 我 认为 我 只 是 提供 了 一 个 合适 的 抽象 容 了 。) 
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错误 。 这 是 因为 此 时 s 实 际 上 没有 与 任何 事物 相关 联 ( 即 ， 没 有 电视 机 )。 因 此 ， 一 种 安全 的 做 
法 是 : 创建 一 个 引用 的 同时 便 进 行 初始 化 。 

String s = "asdf"; 

但 这 里 用 到 了 Java 语 言 的 一 个 特性 : 字符 串 可 以 用 带 引号 的 文本 初始 化 。 通 常 ， 必 须 对 对 
象 采用 一 种 更 通用 的 初始 化 方法 。 


2.2 必须 由 你 创建 所 有 对 象 


一 旦 创建 了 一 个 引用 ， 就 希望 它 能 与 一 个 新 的 对 象 相关 联 。 通 常用 new 操 作 符 来 实现 这 一 目 
的 。new 关键 字 的 意思 是 “给 我 一 个 新 对 象 。” 所 以 前 面 的 例子 可 以 写成 : 

String s = new String("asdf"); 

CRI “ORME TER”, TERME, MHT EREA 
String 的 信息 。 

当然 ， 除 了 String 类 型 ，Java 提 供 了 大 量 过 剩 的 现成 类 型 。 重 要 的 是 ， 你 可 以 自行 创建 类 型 。 
事实 上 ， 这 是 Java 程 序 设计 中 一 项 基本 行为 ， 你 会 在 本 书 以 后 章节 中 慢 慢 学 到 。 

2.2.1 存储 到 什么 地 方 

程序 运行 时 ， 对 象 是 怎么 进行 放置 安排 的 呢 ? 特别 是 内 存 是 怎样 分 配 的 呢 ? 对 这 些 方面 的 
了 解 会 对 你 有 很 大 的 帮助 。 有 五 个 不 同 的 地 方 可 以 存储 数据 ; 

1) 寄存 器 。 这 是 最 快 的 存储 区 ， 因 为 它 位 于 不 同 于 其 他 存储 区 的 地 方 一 -处理 器 内 部 。 但 
是 寄存 器 的 数量 极其 有 限 ， 所 以 寄存 器 根据 需求 进行 分 配 。 你 不 能 直接 控制 ， 也 不 能 在 程序 中 
感觉 到 寄存 器 存在 的 任何 迹象 〈 另 一 方面 ，C 和 C++ 克 许 您 向 编译 器 建议 寄存 器 的 分 配方 式 ) 。 

DHR. TIRAM (随机 访问 存储 器 ) 中 ， 但 通过 堆栈 指针 可 以 从 处 理 器 那里 获得 直 
接 支持 。 堆 栈 指针 车 向 下 移动 ， 则 分 配 新 的 内 存 ， 若 向 上 移动 ， 则 释放 那些 内 存 。 这 是 一 种 快 
速 有 效 的 分 配 存储 方法 ， 仅 次 于 寄存 器 。 创 建 程序 时 ，Java 系 统 必须 知道 存储 在 堆栈 内 所 有 项 
的 确切 生命 周期 ， 以 便 上 下 移动 堆栈 指针 。 这 一 约束 限制 了 程序 的 灵活 性 ， 所 以 虽然 某 些 Java 
数据 存储 于 堆栈 中 一 特别 是 对 象 引用 ， 但 是 Java 对 象 并 不 存储 于 其 中 。 

3) 堆 。 一 种 通用 的 内 存 池 (也 位 于 RAM 区 )， 用 于 存放 所 有 的 Java 对 象 。 堆 不 同 于 堆栈 的 好 
处 是 : 编译 器 不 需要 知道 存储 的 数据 在 堆 里 存活 多 长 时 间 。 因 此 ， 在 堆 里 分 配 存储 有 很 大 的 灵 
活性 。 当 需要 一 个 对 象 时 ， 只 需 用 new 写 一 行 简单 的 代码 ， 当 执行 这 行 代码 时 ， 会 自动 在 堆 里 进 
行 存储 分 配 。 当 然 ， 为 这 种 灵活 性 必须 要 付出 相应 的 代价 ， 用 堆 进行 存储 分 配 和 清理 可 能 比 用 
堆栈 进行 存储 分 配 需 要 更 多 的 时 间 (如 果 确 实 可 以 在 Java 中 像 在 C++ 中 一 样 在 栈 中 创建 对 象 )。 

4) 常量 存储 。 常 量 值 通常 直接 存放 在 程序 代码 内 部 ， 这 样 做 是 安全 的 ， 因 为 它们 永远 不 会 
被 改变 。 有 时 ， 在 戏 入 式 系统 中 ， 常 量 本 身 会 和 其 他 部 分 隔离 开 ， 所 以 在 这 种 情况 下 ， 可 以 先 
择 将 其 存放 在 ROM (只 读 存储 器 ) 中 = 。 

5) 非 RAM 存 储 。 如 果 数 据 完全 存活 于 程序 之 外 ， 那 么 它 可 以 不 受 程序 的 任何 控制 ， 在 程序 
没有 运行 时 也 可 以 存在 。 其 中 两 个 基本 的 例子 是 流 对 象 和 桂 久 化 对 象 。 在 流 对 象 中 ， 对 象 转化 
成 字 节 流 ， 通 常 被 发 送 给 另 一 台 机 器 。 在 “持久 化 对 象 ”中 ， 对 象 被 存放 于 磁盘 上 ， 因 此 ， 即 
使 程序 终止 ， 它 们 仍 可 以 保持 自己 的 状态 。 这 种 存储 方式 的 技巧 在 于 : 把 对 象 转化 成 可 以 存放 





O 这 种 存储 区 的 一 个 例子 是 字符 串 池 。 所 有 字面 常量 字符 串 和 具有 字符 串 值 的 常量 表达 式 都 自动 是 内 存 限 定 的 ， 
并 且 会 置 于 特殊 的 静态 存储 区 中 。 
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在 其 他 媒介 上 的 事物 ， 在 需要 时 ， 可 恢复 成 常规 的 、 基 于 RAM 的 对 象 。Java 提 供 了 对 轻 量 级 持 
久 化 的 支持 ， 而 诸如 JDBC 和 Hibernate 这 样 的 机 制 提供 了 更 加 复杂 的 对 在 数据 库 中 存储 和 读 取 对 
象 信息 的 支持 。 [64] 
2.2.2 特例 : 基本 类 型 
在 程序 设计 中 经 常用 到 一 系列 类 型 ， 它 们 需要 特殊 对 待 。 可 以 把 它们 想像 成 “基本 ”类 型 。 
之 所 以 特殊 对 待 ， 是 因为 new 将 对 象 存储 在 “ 堆 ” 里 ， 故 用 new 创 建 一 个 对 象 一 -特别 是 小 的 、 
简单 的 变量 ， 往 往 不 是 很 有 效 。 因 此 ， 对 于 这 些 类 型 ，Java 采 取 与 C 和 C++ 相同 的 方法 。 也 就 是 
说 ， 不 用 new 来 创建 变量 ， 而 是 创建 一 个 并 非 是 引用 的 “自动 ”变量 。 这 个 变量 直接 存储 “ 值 ”， 
并 置 于 堆栈 中 ， 因 此 更 加 高 效 。 
Java 要 确定 每 种 基本 类 型 所 占 存储 空间 的 大 小 。 它 们 的 大 小 并 不 像 其 他 大 多 数 语言 那样 随 
机 器 硬件 架构 的 变化 而 变化 。 这 种 所 占 存储 空间 大 小 的 不 变性 是 Java 程 序 比 用 其 他 大 多 数 语言 
编写 的 程序 更 具 可 移植 性 的 原因 之 一 。 





基本 类 型 | 大 小 | 最 小 什 最 大 值 包装 器 类 型 
boolean 
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所 有 数值 类 型 都 有 正 负 号 ， 所 以 不 要 去 寻找 无 符号 的 数值 类 型 。 

boolean 类 型 所 占 存储 空间 的 大 小 没有 明确 指定 ， 仅 定义 为 能 够 取 字 面值 true 或 false。 

基本 类 型 具有 的 包装 器 类 ， 使 得 可 以 在 堆 中 创建 一 个 非 基本 对 象 ， 用 来 表示 对 应 的 基本 类 
型 。 例 如 : 

char c = ‘x's 

Character ch = new Character (c); 

也 可 以 这 样 用 ， 

Character ch = new Character (*x'); 

Java SE5 的 自动 包装 功能 将 自动 地 将 基本 类 型 转换 为 包装 器 类 型 : 

Character ch = 'x'; 

并 可 以 反 向 转换 ， 

char c = ch; 

包装 基本 类 型 的 原因 将 在 以 后 的 章节 中 说 明 。 

高 精度 数字 

Java 提 供 了 两 个 用 于 高 精度 计算 的 类 : BigInteger 和 BigDecimal。 虽 然 它们 大 体 上 属于 “ 包 
装 器 类 ”的 范畴 ， 但 二 者 都 没有 对 应 的 基本 类 型 。 

不 过 ， 这 两 个 类 包含 的 方法 ， 提 供 的 操作 与 对 基本 类 型 所 能 执行 的 操作 相似 。 也 就 是 说 ， 
能 作用 于 int 或 float 的 操作 ， 也 同样 能 作用 于 BigInteger 或 BigDecimal。 只 不 过 必须 以 方法 调用 
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方式 取代 运算 符 方式 来 实现 。 由 于 这 么 做 复杂 了 许多 ， 所 以 运算 速度 会 比较 慢 。 在 这 里 ， 我 们 
以 速度 换取 了 精度 。 

BigInteger 支 持 任意 精度 的 整数 。 也 就 是 说 ， 在 运算 中 ,可 以 准确 地 表示 任何 大 小 的 整数 值 ， 
而 不 会 丢失 任何 信息 。 

BigDecimal 支 持 任何 精度 的 定点 数 ， 例 如 ， 可 以 用 它 进行 精确 的 货币 计算 。 

关于 调用 这 两 个 类 的 构造 器 和 方法 的 详细 信息 ， 请 查阅 JDK 文 档 。 

2.2.3 Java 中 的 数组 

几乎 所 有 的 程序 设计 语言 都 支持 数组 。 在 C 和 C++ 中 使 用 数组 是 很 危险 的 ， 因 为 C 和 C++ 中 
的 数组 就 是 内 存 块 。 如 果 一 个 程序 要 访问 其 自身 内 存 块 之 外 的 数组 ， 或 在 数组 初始 化 前 使 用 内 
存 (程序 中 常见 的 错误 ) ， 都 会 产生 难以 预料 的 后 果 。 

Java 的 主要 目标 之 一 是 安全 性 ， 所 以 许多 在 C 和 C++ 里 困扰 程序 员 的 问题 在 Java 里 不 会 再 出 
现 。Java 确 保 数组 会 被 初始 化 ， 而 且 不 能 在 它 的 范围 之 外 被 访问 。 这 种 范围 检查 ， 是 以 每 个 数 
组 上 少量 的 内 存 开销 及 运行 时 的 下 标 检查 为 代价 的 。 但 由 此 换 来 的 是 安全 性 和 效率 的 提高 ， 
此 付出 的 代价 是 值得 的 (并 且 Java 有 时 可 以 优化 这 些 操作 )。 

当 创 建 一 个 数组 对 象 时 ， 实 际 上 就 是 创建 了 一 个 引用 数组 ， 并 且 每 个 引用 都 会 自动 被 初始 
化 为 一 个 特定 值 ， 该 值 拥 有 自己 的 关键 字 null。 一 旦 Java 看 到 null， 就 知道 这 个 引用 还 没有 指向 
某 个 对 象 。 在 使 用 任何 引用 前 ， 必 须 为 其 指定 一 个 对 象 ， 如 果 试 图 使 用 一 个 还 是 null 的 引用 ， 在 
运行 时 将 会 报错 。 因 此 ， 常 犯 的 数组 错误 在 Java 中 就 可 以 避免 。 

还 可 以 创建 用 来 存放 基本 数据 类 型 的 数组 。 同 样 ， 编 译 器 也 能 确保 这 种 数组 的 初始 化 ， 因 
为 它 会 将 这 种 数组 所 占 的 内 存 全 部 置 零 。 

数组 将 在 以 后 的 章节 中 详细 讨论 。 


2.3 永远 不 需要 销毁 对 象 - 


在 大 多 数 程序 设计 语言 中 ， 变 量 生 命 周 期 的 概念 ， 占 据 了 程序 设计 工作 中 非常 重要 的 部 分 。 
变量 需要 存活 多 长 时 间 ? 如 果 想 要 销毁 对 象 ， 那 什么 时 刻 进 行 呢 ? 变量 生命 周期 的 混乱 往往 会 
导致 大 量 的 程序 bug， 本 节 将 介绍 Java 是 怎样 替 我 们 完成 所 有 的 清理 工作 ， 从 而 大 大 地 简化 这 个 
问题 的 。 

2.3.1 作用 域 

大 多 数 过 程 型 语言 都 有 作用 域 (scope) 的 概念 。 作 用 域 决定 了 在 其 内 定义 的 变量 名 的 可 见 

性 和 生命 周期 。 在 C、C++ 和 Java 中 ， 作 用 域 由 花 括号 的 位 置 决定 。 例 如 ， 


{ 
int x = 12; 
// Only x available 


int q = 96; 
J/ Both x & q available 


} 

// Only x available 

/1/ q is “out of scope” 
} 


在 作用 域 里 定义 的 变量 只 可 用 于 作用 域 结束 之 前 。 

任何 位 于 “/” 之 后 到 行 末 的 文字 都 是 注释 。 

缩 排 格式 使 Iava 代 码 更 易于 阅读 。 由 于 Java 是 一 种 自由 格式 (free-form) 的 语言 ， 所 以 ， 空 
格 、 制 表 符 、 换 行 都 不 会 影响 程序 的 执行 结果 。 
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尽管 以 下 代码 在 C 和 C++ 中 是 合法 的 ， 但 是 在 Java 中 却 不 能 这 样 书写 
{nt xe = 12; 


{ 


} 
} 


编译 器 将 会 报告 变量 x 已 经 定义 过 。 所 以 ， 在 C 和 C++ 里 将 一 个 较 大 作用 域 的 变量 “隐藏 
起 来 的 做 法 ， 在 Java 里 是 不 允许 的 。 因 为 Java 设 计 者 认为 这 样 做 会 导致 程序 混乱 。 
2.3.2 对 象 的 作用 域 

Java 对 象 不 具备 和 基本 类 型 一 样 的 生命 周期 。 当 用 new 创建 一 个 Java 对 象 时 ， 它 可 以 存活 于 
作用 域 之 外 。 所 以 假如 你 采用 代码 

{ 


String s = new String("a string"); 
} // End of scope 


引用 s 在 作用 域 终点 就 消失 了 。 然 而 ，s 指 向 的 String 对 象 仍 继续 占据 内 存 空 间 。 在 这 一 小 段 代码 
中 ， 我 们 无 法 在 这 个 作用 域 之 后 访问 这 个 对 象 ， 因 为 对 它 唯一 的 引用 已 超出 了 作用 域 的 范围 
在 后 继 章 节 中 ， 读 者 将 会 看 到 : 在 程序 执行 过 程 中 ;怎样 传递 和 复制 对 象 引用 。 

事实 证 明 ， 由 new 创 建 的 对 象 ， 只 要 你 需要 ， 就 会 一 直 保 留 下 去 。 这 样 ， 许 多 C++ 编 程 问题 
在 Java 中 就 完全 消失 了 。 在 C++ 中 ， 你 不 仅 必须 要 确保 对 象 的 保留 时 间 与 你 需要 这 些 对 象 的 时 间 
一 样 长 ， 而 且 还 必须 在 你 使 用 完 它们 之 后 ， 将 其 销毁 。 

这 样 便 带 来 一 个 有 趣 的 问题 。 如 果 Java 让 对 象 继续 存在 ， 那 么 靠 什么 才能 防止 这 些 对 象 填 
满 内 存 空间 ， 进 而 阻塞 你 的 程序 昵 ? 这 正 是 C++ 里 可 能 会 发 生 的 问题 。 这 也 是 Java 神 奇 之 所 在 。 
Java 有 一 个 垃圾 回收 器 ， 用 来 监视 用 new 创建 的 所 有 对 象 ， 并 辨别 那些 不 会 再 被 引用 的 对 象 。 随 
后 ， 释 放 这 些 对 象 的 内 存 空间 ， 以 便 供 其 他 新 的 对 象 使 用 。 也 就 是 说 ， 你 根本 不 必 担心 内 存 回 
收 的 问题 。 你 只 需要 创建 对 象 ， 一 旦 不 再 需要 ， 它 们 就 会 自行 消失 。 这 样 做 就 消除 了 这 类 编程 
问题 ( 即 “ 内 存 泄漏 ")， 这 是 由 于 程序 员 忘 记 释 放 内 存 而 产生 的 问题 。 


2.4 创建 新 的 数据 类 型 : 类 


如 果 一 切 都 是 对 象 ， 那 么 是 什么 决定 了 某 一 类 对 象 的 外 观 与 行为 呢 ? 换 名 话说， 是 什么 确 
定 了 对 象 的 类 型 ? 你 可 能 期 望 有 一 个 名 为 “type” 的 关键 字 ， 当 然 它 必 须 还 要 有 相应 的 含义 。 然 
和 而， 从 历史 发 展 角度 来 看 ， 大 多 数 面向 对 象 的 程序 设计 语言 习惯 用 关键 字 class 来 表示 “我 准备 
告诉 你 一 种 新 类 型 的 对 象 看 起 来 像 什么 样子 ” 。class 这 个 关键 字 (以 后 会 频繁 使 用 ， 本 书 以 后 就 
不 再 用 粗 体 字 表示 ) 之 后 紧 跟着 的 是 新 类 型 的 名 称 。 例 如 : 


class ATypeName { /* Class body goes here */ } 


这 就 引入 了 一 种 新 的 类 型 ， 尽 管 类 主体 仅 包含 一 条 注释 语句 〈 星 号 和 斜 杠 以 及 其 中 的 内 容 就 是 
注释 ， 本 章 后 面 再 讨论 )。 因 此 ， 你 还 不 能 用 它 做 太 多 的 事情 。 然 而 ， 你 已 经 可 以 用 new 来 创建 
这 种 类 型 的 对 象 : 


ATypeName a = new ATypeName() ; 


但 是 ， 在 定义 它 的 所 有 方法 之 前 ， 还 没有 办 法 能 让 它 去 做 更 多 的 事情 (也 就 是 说 ， 不 能 向 它 发 
送 任何 有 意义 的 消息 )。 


int x = 96; /7 Illegal 
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2.4.1 字段 和 方法 > 
一 旦 定义 了 一 个 类 (在 Java 中 你 所 做 的 全 部 工作 就 是 定义 类 ， 产 生 那 些 类 的 对 象 ， 以 及 发 
送 消息 给 这 些 对 象 )， 就 可 以 在 类 中 设置 两 种 类 型 的 元 素 : 字段 (有 了 时 被 称 作 数 据 成 员 ) 和 方法 
(有 时 被 称 作成 员 函 数 )。 字 段 可 以 是 任何 类 型 的 对 象 ， 可 以 通过 其 引用 与 其 进行 通信 ， 也 可 以 
是 基本 类 型 中 的 一 种 。 如 果 字 段 是 对 某 个 对 象 的 引用 ， 那 么 必须 初始 化 该 引用 ， 以 便 使 其 与 一 
个 实际 的 对 象 (如 前 所 述 ， 使 用 new 来 实现 ) 相关 联 。 
每 个 对 象 都 有 用 来 存储 其 字段 的 空间 ， 普 通 字段 不 能 在 对 象 间 共 享 。 下 面 是 一 个 具有 某 些 
字段 的 类 ， 
class DataOnly { 
int i; 
double d; 


boolean b; 
} 


尽管 这 个 类 除了 存储 数据 之 外 什么 也 不 能 做 ， 但 是 仍旧 可 以 像 下 面 这 样 创建 它 的 一 个 对 象 ; 

DataOnly data = new DataOnly(); 

可 以 给 字段 赋值 ， 但 首先 必须 知道 如 何 引用 一 个 对 象 的 成 员 。 具 体 的 实现 为 ;在 对 象 引用 
的 名 称 之 后 紧 接着 一 个 句点 ， 然 后 再 接着 是 对 象 内 部 的 成 员 名 称 : ` 

objectReference.member : 

例如 ， 


data.i = 47; 
data.d = 1.1; 
data.b = false; 


想 修改 的 数据 也 有 可 能 位 于 对 象 所 包含 的 其 他 对 象 中 。 在 这 种 情况 下 ， 只 需要 再 使 用 连接 
句点 即 可 。 例 如 : 

myPlane. leftTank. capacity = 108; 

DataOnly 类 除了 保存 数据 外 没 别 的 用 处 ， 因 为 它 没有 任何 成 员 方法 。 如 果 想 了 解 成 员 方法 
的 运行 机 制 ， 就 得 先 了 解 参 数 和 返回 值 的 概念 ， 稍 后 将 对 此 作 简 略 描述 。 

基本 成 员 默 认 值 

若 类 的 某 个 成 员 是 基本 数据 类 型 ,即使 没有 进行 初始 化 ， 
Java 也 会 确保 它 获得 一 个 默认 值 ， 如 右 表 所 示 : 
当 变 量 作为 类 的 成 员 使 用 时 ，Java 才 确保 给 定 其 默认 值 ， 以 
确保 那些 是 基本 类 型 的 成 员 变量 得 到 初始 化 (C++ 没有 此 功 
能 )， 防止 产 生 程序 错误 。 但 是 ， 这 些 初始 值 对 你 的 程序 来 
说 ， 可 能 是 不 正确 的 ， 其 至 是 不 合法 的 。 所 以 最 好 明确 地 对 
变量 进行 初始 化 。 f 

然而 上 述 确保 初始 化 的 方法 并 不 适用 于 “局 部 ”变量 〈 即 并 非 某 个 类 的 字段 )。 因 此 ， 如 果 
在 某 个 方法 定义 中 有 

int x: 
那么 变量 x 得 到 的 可 能 是 任意 值 (SC 和 C++ 中 一 样 ) ， 而 不 会 被 自动 初始 化 为 零 。 所 以 在 使 用 x 
前 ， 应 先 对 其 赋 一 个 适当 的 值 。 如 果 忘 记 了 这 么 做 ，Java 会 在 编译 时 返回 一 个 错误 ， 告 诉 你 此 
变量 没有 初始 化 ， 这 正 是 Java 优 于 C++ 的 地 方 。( 许 多 C++ 编译 器 会 对 未 初始 化 变量 给 予 警 告 ， 
而 Java 则 视 为 是 错误 )。 





默认 值 







| 
Yaoooo' (null) 
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25 方法 、 参 数 和 返回 值 


许多 程序 设计 语言 ( 像 C 和 C++) 用 函数 这 个 术语 来 描述 命名 子 程序 ， 而 在 Java 里 却 常用 方 
法 这 个 术语 来 表示 “做 某 些 事情 的 方式 ”。 实 际 上 ， 继 续 把 它 看 作 是 函数 也 无 妨 。 尽 管 这 只 是 用 
词 上 的 差别 ， 但 本 书 将 沿用 Java 的 惯用 法 ， 即 用 术语 “方法 ”而 不 是 “函数 ”来 描述 。 

Java 的 方法 决定 了 一 个 对 象 能 够 接收 什么 样 的 消息 。 方 法 的 基本 组 成 部 分 包括 : 名 称 、 参 
数 、 返 回 值 和 方法 体 。 下 面 是 它 最 基本 的 形式 : 


ReturnType methodName( /* Argument list */ ) { 
/* Method body */ 
} 


返回 类 型 描述 的 是 在 调用 方法 之 后 从 方法 返回 的 值 。 参 数列 表 给 出 了 要 传 给 方法 的 信息 的 
类 型 和 名 称 。 方 法 名 和 参数 列表 (它们 合 起 来 被 称 为 “方法 签名 ”") 唯一 地 标识 出 某 个 方法 。 

Java 中 的 方法 只 能 作为 类 的 一 部 分 来 创建 。 方 法 只 有 通过 对 象 才能 被 调用 ， 且 这 个 对 象 必 
须 能 执行 这 个 方法 调用 。 如 果 试图 在 某 个 对 象 上 调用 它 并 不 具备 的 方法 ， 那 么 在 编译 时 就 会 得 
到 一 条 错误 消息 。 通 过 对 象 调用 方法 时 ， 需 要 先 列 出 对 象 名 ， 紧 接着 是 句点 ， 然 后 是 方法 名 和 
参数 列表 。 如 ， 

objectName.methodName(argi, arg2, arg3): 

例如 ， 假 设 有 一 个 方法 f0， 不 带 任何 参数 ， 返 回 类 型 是 int。 如 果 有 个 名 为 8 的 对 象 、 可 以 通 
过 它 调用 f0， 那 么 就 可 以 这 样 写 : 

int x = a.f0; 

返回 值 的 类 型 必须 要 与 x 的 类 型 兼容 。 

这 种 调用 方法 的 行为 通常 被 称 为 发 送 消息 给 对 象 。 在 上 面 的 例子 中 ， 消 息 是 fD0， 对 象 是 a。 
面向 对 象 的 程序 设计 通常 简单 地 归纳 为 “向 对 象 发 送 消息 "。 
2.5.1 参数 列表 

方法 的 参数 列表 指定 要 传递 给 方法 什么 样 的 信息 。 正 如 你 可 能 料想 的 那样 ， 这 些 信息 像 
Java 中 的 其 他 信息 一 样 ， 采 用 的 都 是 对 象形 式 。 因 此 ， 在 参数 列表 中 必须 指定 每 个 所 传递 对 象 
的 类 型 及 名 字 。 像 Java 中 任何 传递 对 象 的 场合 一 样 ， 这 里 传递 的 实际 上 也 是 引用 ， 并 且 引 用 的 
类 型 必须 正确 。 如 果 参 数 被 设 为 String 类 型 ， 则 必须 传递 一 个 String 对 象 ， 否 则 ， 编 译 器 将 抛 出 
错误 。 

假设 某 个 方法 接受 String 为 其 参数 ， 下 面 是 其 具体 定义 ， 它 必须 置 于 某 个 类 的 定义 内 才能 被 
正确 编译 。 


int storage(String s) { 
return s.length() * 2; 
} 


此 方法 告诉 你 ， 需 要 多 少 个 字 节 才能 容纳 一 个 特定 的 String 对 象 中 的 信息 (字符 串 中 的 每 个 
字符 的 尺寸 都 是 16 位 或 2 个 字 节 ， 以 此 来 提供 对 Unicode 字 符 集 的 支持 ) 。 此 方法 的 参数 类 型 是 
String， 参 数 名 是 s。 一 旦 将 s 传 递 给 此 方法 ， 就 可 以 把 他 当 作 其 他 对 象 一 样 进行 处 理 ( 可 以 给 它 
传递 消息 )。 在 这 里 ，s 的 length0 方 法 被 调用 ， 它 是 String 类 提供 的 方法 之 一 ， 会 返回 字符 串 包 


O 稍 后 将 会 学 到 static 方 法 ， 它 是 针对 类 调用 的 ， 并 不 依赖 于 对 象 的 存在 。 
O 对 于 前 面 所 提 到 的 特殊 数据 类 型 boolean、char、byte、short、int、long、float 和 double 来 说 是 一 个 例外 。 通 
常 ， 尽 管 传递 的 是 对 象 ， 而 实际 上 传递 的 是 对 象 的 引用 。 
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含 的 字符 数 。 

通过 上 面 的 例子 ， 还 可 以 了 解 到 return 关 键 字 的 用 法 ， 它 包括 两 方面 : 首先 ， 它 代表 “已 经 
做 完 ， 离 开 此 方法 ”。 其 次 ， 如 果 此 方法 产生 了 一 个 值 ， 这 个 值 要 放 在 return 语 名 后面。 在 这 个 
例子 中 ， 返 回 值 是 通过 计算 s.length0 * 2 这 个 表达 式 得 到 的 。 

你 可 以 定义 方法 返回 任意 想 要 的 类 型 ， 如 果 不 想 返 回 任何 值 ， 可 以 指示 此 方法 返回 void 
( 空 )。 下 面 是 一 些 例子 : 


boolean flag() { return true; } 

double naturalLogBase() { return 2.718; } 
void nothing() { return; } 

void nothing2() {} 


车 返回 类 型 是 void，return 关 键 字 的 作用 只 是 用 来 退出 方法 。 因 此 ， 没 有 必要 到 方法 结束 时 
才 离 开 ， 可 在 任何 地 方 返回 。 但 如 果 返 回 类 型 不 是 void， 那 么 无 论 在 何 处 返回 ， 编 译 器 都 会 强 
制 返回 一 个 正确 类 型 的 返回 值 。 

到 此 为 止 ， 读 者 或 许 觉得 : 程序 似乎 只 是 一 系列 带 有 方法 的 对 象 组 合 ， 这 些 方法 以 其 他 对 
象 为 参数 ， 并 发 送 消息 给 其 他 对 象 。 大 体 上 确实 是 这 样 ， 但 在 以 后 章节 中 ， 读 者 将 会 学 到 怎样 
在 一 个 方法 内 进行 判断 ， 做 一 些 更 细致 的 底层 工作 。 至 于 本 章 ， 读 者 只 需要 理解 消息 发 送 就 足 
够 了 。 


2.6 构建 一 个 Java 程 序 


在 构建 自己 的 第 一 个 Java 程 序 前 ， 还 必须 了 解 其 他 一 些 问 题 。 
2.6.1 名 字 可 见 性 

名 字 管 理 对 任何 程序 设计 语言 来 说 ， 都 是 一 个 重要 问题 。 如 果 在 程序 的 某 个 模块 里 使 用 了 
一 个 名 字 ， 而 其 他 人 在 这 个 程序 的 另 一 个 模块 里 也 使 用 了 相同 的 名 字 ， 那 么 怎样 才能 区 分 这 两 
个 名 字 并 防止 二 者 互相 冲突 呢 ? 这 个 问题 在 C 语 言 中 尤其 严重 ， 因 为 程序 往往 包含 许多 难以 管理 
的 名 字 。C++ 类 (Java 类 基于 此 ) 将 函数 包 于 其 内 ， 从 而 避免 了 与 其 他 类 中 的 函数 名 相 冲 突 。 然 
而 ，C++ 仍 允许 全 局 数据 和 全 局 函数 的 存在 ， 所 以 还 是 有 可 能 发 生 冲 突 。 为 了 解决 这 个 问题 ， 
C++ 通过 几 个 关键 字 引 入 了 名 字 空 间 的 概念 。 

Java 采 用 了 一 种 全 新 的 方法 来 避免 上 述 所 有 问题 。 为 了 给 一 个 类 库 生 成 不 会 与 其 他 名 字 混 清 
的 名 字 ，Java 设 计 者 希望 程序 员 反 过 来 使 用 自己 的 Intemet 域 名 ， 因 为 这 样 可 以 保证 它们 肯定 是 独 
一 无 二 的 。 由 于 我 的 域名 是 MindView.net， 所 以 我 的 各 种 奇 奇怪 怪 的 应 用 工具 类 库 就 被 命名 为 
net.mindview.utility.foibles。 反 转 域名 后 ， 句 点 就 用 来 代表 子 目 录 的 划分 。 

在 Java 1.0 和 Java 1.1 中 ， 扩展 名 com、edu、org、net 等 约定 为 大 写 形式 。 所 以 上 面 的 库 名 应 
该 写成 NET.mindview.utility.foibles。 然 而 ， 在 Java 2 开发 到 一 半 时 ， 设 计 者 们 发 现 这 样 做 会 引起 
一 些 问题 ， 因 此 ， 现 在 整个 包 名 都 是 小 写 了 。 

这 种 机 制 意味 着 所 有 的 文件 都 能 够 自动 存活 于 它们 自己 的 名 字 空 间 内 ， 而 且 同 一 个 文件 内 
的 每 个 类 都 有 唯一 的 标识 符 一 Java 语 言 本 身 已 经 解决 了 这 个 问题 。 
2.6.2 运用 其 他 构件 

如 果 想 在 自己 的 程序 里 使 用 预先 定义 好 的 类 ， 那 么 编译 器 就 必须 知道 怎么 定位 它们 。 当 然 ， 
这 个 类 可 能 就 在 发 出 调用 的 那个 源 文件 中 ， 在 这 种 情况 下 ， 就 可 以 直接 使 用 这 个 类 一 一 即使 这 
个 类 在 文件 的 后 面 才 会 被 定义 (Java 消 除了 所 谓 的 “向 前 引用 ”问题 )。 

如 果 那 个 类 位 于 其 他 文件 中 ， 又 会 怎样 呢 ? 你 可 能 会 认为 编译 器 应 该 有 足够 的 智慧 ， 能 够 
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直接 找到 它 的 位 置 ， 但 事实 并 非 如 此 。 想 像 下 面 的 情况 ， 如 果 你 想 使 用 某 个 特定 名 字 的 类 ， 但 
其 定义 却 不 止 一 份 〈 假 设 这 些 定义 各 不 相同 )。 更 精 糕 的 是 ， 假 设 你 正在 写 一 个 程序 ， 在 构建 过 
程 中 ， 你 想 将 某 个 新 类 添加 到 类 库 中 ， 但 却 与 已 有 的 某 个 类 名 冲突 。 

为 了 解决 这 个 问题 ， 必 须 消除 所 有 可 能 的 混 清 情况。 为 实现 这 个 目的 ， 可 以 使 用 关键 字 
import 来 准确 地 告诉 编译 器 你 想 要 的 类 是 什么 。import 指 示 编译 器 导入 一 个 包 ， 也 就 是 一 个 类 
E (在 其 他 语言 中 ， 一 个 库 不 仅 包含 类 ， 还 可 能 包括 方法 和 数据 :但 是 Java 中 所 有 的 代码 都 必 
须 写 在 类 里 )。 

大 多 时 候 ， 我 们 使 用 与 编译 器 附 在 一 起 的 Java 标 准 类 库 里 的 构件 。 有 了 这 些 构件 ， 你 就 不 
必 写 一 长 串 的 反 转 域名 。 举 例 来 说 ， 只 须 像 下 面 这 么 书写 就 行 了 : 


import java.util.ArrayList; 


这 行 代码 告诉 编译 器 ， 你 想 使 用 Java 的 ArrayList 类 。 但 是 ， wee THREE, 有 时 
你 想 使 用 其 中 的 几 个 ， 同 时 又 不 想 明确 地 逐一 声明 ， 那 么 你 很 容易 使 用 通配符 “*” 来 达到 这 个 
GESE 

import java.util.*; 

这 种 一 次 导 人 一 群 类 的 方式 比 一 个 一 个 地 导入 类 的 方式 更 常用 。 
2.6.3 static 关键 字 

通常 来 说 ， 当 创建 类 时 ， 就 是 在 描述 那个 类 的 对 象 的 外 观 与 行为 。 除 非 用 new 创 建 那个 类 的 
对 象 ， 否 则 ， 实 际 上 并 未 获得 任何 对 象 。 执 行 new 来 创建 对 象 时 ， 数 据 存储 空间 才 被 分 配 ， 其 方 
法 才 供 外 界 调用 。 

有 两 种 情形 用 上 述 方法 是 无 法 解决 的 ;一 种 情形 是 ， 只 想 为 某 特定 域 分 配 单一 存储 空间 ， 
而 不 去 考虑 究竟 要 创建 多 少 对 象 ， 甚 至 根本 就 不 创建 任何 对 象 。 另 一 种 情形 是 ， 希 望 某 个 方法 
不 与 包含 它 的 类 的 任何 对 象 关 联 在 一 起 。 也 就 是 说 ， 即 使 没有 创建 对 象 ， 也 能 够 调用 这 个 方法 。 

通过 static 关 键 字 可 以 满足 这 两 方面 的 需要 。 当 声明 一 个 事物 是 static 时 ， 就 意味 着 这 个 域 或 
方法 不 会 与 包含 它 的 那个 类 的 任何 对 象 实例 关联 在 一 起 。 所 以 ， 即 使 从 未 创建 某 个 类 的 任何 对 
象 ， 也 可 以 调用 其 static 方 法 或 访问 其 static 域 。 通 常 ， 你 必须 创建 一 个 对 象 ， 并 用 它 来 访问 数据 
或 方法 。 因 为 非 static 域 和 方法 必须 知道 它们 一 起 运作 的 特定 对 象 。 。 

有 些 面向 对 象 语言 采用 类 数据 和 类 方法 两 个 术语 ， 代 表 那 些 数据 和 方法 只 是 作为 整个 类 ， 
而 不 是 类 的 某 个 特定 对 象 而 存在 的 。 有 了 时， 一 些 Java 文 献 里 也 用 到 这 两 个 术语 。 

只 须 将 static 关 键 字 放 在 定义 之 前 ， 就 可 以 将 字段 或 方法 设 定 为 static。 例 如 ， 下 面 的 代码 就 
生成 了 一 个 static 字 段 ， 并 对 其 进行 了 初始 化 : 


class StaticTest { 
static int i = 47; 
} 


现在 ， 即 使 你 创建 了 两 个 StaticTest 对 象 ，StaticTest.i 也 只 有 一 份 存储 空间 ， 这 两 个 对 象 共 
享 同一 个 i。 再 看 看 下 面 代码 : 


StaticTest stl = new StaticTest(); 
StaticTest st2 = new StaticTest(); 


在 这 里 ，stLi 和 st2.i 指 向 同一 存储 空间 ， 因 此 它们 具有 相同 的 值 48。 
引用 static 变 量 有 两 种 方法 。 如 前 例 所 示 ， 可 以 通过 一 个 对 象 去 定位 它 ， 如 st2.i; 也 可 以 通 


O 当然 ， 由 于 在 用 static 方 法 前 不 需要 创建 任何 对 象 ， 所 以 对 于 static 方 法 ， 不 能 简单 地 通过 调用 其 他 非 static 域 或 
方法 而 没有 指定 某 个 命名 对 象 , 来 直接 访问 非 static 域 或 方法 (因为 非 static 域 或 方法 必须 与 某 一 特定 对 象 关联 )。 
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过 其 类 名 直接 引用 ， 而 这 对 于 非 静态 成 员 则 不 行 。 


StaticTest.it+; 


其 中 ，++ 运 算 符 对 变量 进行 递 加 操作 。 此 时 ，stli 和 st2.i 仍 具有 相同 的 值 48。 

使 用 类 名 是 引用 static 变 量 的 首选 方式 ， 这 不 仅 是 因为 它 强调 了 变量 的 static 结 构 ， 而 且 在 某 
些 情况 下 它 还 为 编译 器 进行 优化 提供 了 更 好 的 机 会 。 

类 似 逻 辑 也 应 用 于 静态 方法 。 既 可 以 像 其 他 方法 一 样 ， 通 过 一 个 对 象 来 引用 某 个 静态 方法 ， 
也 可 以 通过 特殊 的 语法 形式 ClassName.method0 加 以 引用 。 定 义 静态 方法 的 方式 也 与 定义 静态 
变量 的 方式 相似 ; 


class Incrementable { 
static void increment() { StaticTest.i++; } 
} 


可 以 看 到 ，Inerementable 的 increment0 方 法 通过 ++ 运 算 符 将 静态 数据 i 递 加 。 可 以 采用 典 
型 的 方式 ， 通 过 对 象 来 调用 increment0 : 


Incrementable sf = new Incrementable(); 
sf.increment(); 


或 者 ， 因 为 inerement0 是 一 个 静态 方法 ， 所 以 也 可 以 通过 它 的 类 直接 调用 ， 


Incrementable. increment(); 


尽管 当 statie 作 用 于 某 个 字段 时 ， 肯 定 会 改变 数据 创建 的 方式 (因为 一 个 static 字 段 对 每 个 类 
来 说 都 只 有 一 份 存储 空间 ， 而 非 static 字 段 则 是 对 每 个 对 象 有 一 个 存储 空间 ) ， 但 是 如 果 static 作 
用 于 某 个 方法 ， 差 别 却 没有 那么 大 。static 方 法 的 一 个 重要 用 法 就 是 在 不 创建 任何 对 象 的 前 提 下 
就 可 以 调用 它 。 正 如 我 们 将 会 看 到 的 那样 ， 这 一 点 对 定义 main0 方 法 很 重要 ， 这 个 方法 是 运行 
一 个 应 用 时 的 人 口 点 。 

和 其 他 任何 方法 一 样 ，static 方 法 可 以 创建 或 使 用 与 其 类 型 相同 的 被 命名 对 象 ， 因 此 ，static 
方法 常常 拿 来 做 “牧羊 人 ”的 角色 ， 负 责 看 护 与 其 隶属 同一 类 型 的 实例 群 。 


2.7 你 的 第 一 个 Java 程 序 


最 后 ， 让 我 们 编写 第 一 个 完整 的 程序 。 此 程序 开始 是 打印 一 个 字符 串 ， 然 后 是 打印 当前 日 
期 ， 这 里 用 到 了 Java 标 准 库 里 的 Date 类 。 

// HelloDate. java 

import java.util.*: 


public class HelloDate { 
public static void main(String{] args) { 
System.out.printin("Hello, it's: "); 
System.out.printin(new Date()): 
$ 
} 


在 每 个 程序 文件 的 开头 ， 必 须 声 明 import 语 句 ， 以 便 引 入 在 文件 代码 中 需要 用 到 的 额外 类 。 
注意 ， 在 这 里 说 它们 “额外 " ， 是 因为 有 一 个 特定 类 会 自动 被 导入 到 每 一 个 Java 文 件 中 : 
javalang。 打 开 你 的 Web 浏 览 器 ， 查 找 Sun 公 司 提供 的 文档 ( 若 没 有 从 http://iava.sun.com。 F 
JDK 文 档 ， 现 在 开始 下 载 。 注 意 ， 这 个 文档 并 没有 随 JDK 一 起 打包 ， 你 必须 专门 去 下 载 它 )。 在 
包 列 表 里 ， 可 以 看 到 Java 配 套 提供 的 各 种 类 库 。 请 点 击 其 中 的 javalang， 就 会 显示 出 这 个 类 库 所 


O Sun 提 供 的 Java 编 译 器 和 文档 总 是 在 定期 地 变化 ， 所 以 获取 它们 的 最 佳 方式 就 是 直接 从 Sun 处 获得 。 通 过 自己 
下 载 这 些 资料 ， 你 可 以 获得 最 新 的 版 本 。 
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包含 的 全 部 类 的 列表 。 由 于 java.lang 是 默认 导入 到 每 个 Java 文 件 中 的 ， 所 以 它 的 所 有 类 都 可 以 被 
直接 使 用 。java.lang 里 没有 Date 类 ， 所 以 必须 导入 另外 一 个 类 库 才 能 使 用 它 。 若 不 知 某 个 特定 
类 在 哪个 类 库 里 ， 可 在 Java 文 档 中 选择 “Tree”， 便 可 以 看 到 Java 配 套 提供 的 每 一 个 类 。 接 下 来 ， 
用 浏览 器 的 “查找 ”功能 查找 Date。 这 样 就 可 以 发 现 它 以 java.util.Date 的 形式 被 列 了 出 来 。 于 
是 我 们 知道 它 位 于 util 类 库 中 ， 并 且 必 须 书写 import java.util.* 才能 使 用 Date 类 。 

现在 返回 文档 最 开头 的 部 分 ， 选 择 java.lang， 接 着 是 system， 可 以 看 到 system 类 有 许多 属 
性 ， 若 选择 out， 就 会 发 现 它 是 一 个 静态 PrintStream 对 象 。 因 为 是 静态 的 ， 所 以 不 需要 创建 任何 
东西 ，out 对 象 便 已 经 存在 了 ， 只 须 直接 使 用 即 可 。 但 我 们 能 够 用 out 对 象 做 些 什么 事情 ， 是 由 它 
的 类 型 PrintStream 决 定 的 。PrintStream 在 描述 文档 中 是 以 超 链 接 形式 显示 ， 所 以 很 方便 进行 查 
看 ， 只 须 点 击 它 ， 就 可 以 看 到 能 够 为 PrintStream 调 用 的 所 有 方法 。 方 法 的 数量 不 少 ， 本 书后 面 再 
详 加 讨论 。 现 在 我 们 只 对 printin0 方 法 感 兴趣 ， 它 的 实际 作用 是 “将 我 给 你 的 数据 打印 到 控制 台 ， 
完成 后 换行 "。 因 此 ， 在 任何 Java 程 序 中 ， 一 旦 需要 将 某 些 数据 打印 到 控制 台 ， 就 可 以 这 样 写 ; 

System.out.printin("A String of things"); 

类 的 名 字 必 须 和 文件 名 相同 。 如 果 你 像 现 在 这 样 创建 一 个 独立 运行 的 程序 ， 那 么 文件 中 必 
须 存在 某 个 类 与 该 文件 同名 (否则 ， 编 译 器 会 报错 )， 且 那个 类 必须 包含 一 个 名 为 main0 的 方法 ， 
形式 如 下 所 示 ， 

public static void main(String[] args) { 
其 中 ，public 关 键 字 意 指 这 是 一 个 可 由 外 部 调用 的 方法 (第 5 章 将 详细 描述 )。main0 方 法 的 参数 
是 一 个 String 对 象 的 数组 。 在 这 个 程序 中 并 未 用 到 args， 但 是 Java 编 译 器 要 求 必须 这 样 做 ， 因 为 
args 要 用 来 存储 命令 行 参数 。 

打印 日 期 的 这 行 代 码 很 是 有 趣 的 : 

System.out.println(new Date()); 
在 这 里 ， 传 递 的 参数 是 一 个 Date 对 象 ， 一 旦 创建 它 之 后 ， 就 可 以 直接 将 它 的 值 ( 它 被 自动 转换 
为 String 类 型 ) 发 送 给 printin0。 当 这 条 语句 执行 完毕 后 ，Date 对 象 就 不 再 被 使 用 ， 而 垃圾 回收 
器 会 发 现 这 一 情况 ， 并 在 任意 时 候 将 其 回收 。 因 此 ， 我 们 就 没 必 要 去 关心 怎样 清理 它 了 。 

当 你 阅读 从 httpWiava.sun.com 下 载 的 JDK 文 档 时 ， 将 会 发 现 System 有 许多 其 他 的 方法 ， 使 得 
你 可 以 去 创造 很 多 有 趣 的 效果 (Java 最 强大 的 优势 之 一 就 是 它 具有 庞大 的 标准 类 库 集 )。 例 如 : 


//: object/ShowProperties. java 


public class ShowProperties { 
public static void main(String[] args) { 
System. getProperties().list (System. out); 
System. out .printin(System. getProperty("user.name")); 
System. out .printin( 
System. getProperty("java.library.path")); 


} 

duh 

main0 的 第 一 行将 显示 从 运行 程序 的 系统 中 获取 的 所 有 “属性 "， 因 此 它 可 以 向 你 提供 环境 
信息 。listO 方 法 将 结果 发 送 给 它 的 参数 : System.out。 在 本 书后 面 的 章节 中 你 将 会 看 到 ， 你 可 以 
把 结果 发 送 到 任何 地 方 ， 例 如 发 送 到 文件 中 。 你 还 可 以 询问 具体 的 属性 ， 例 如 在 本 例 中 ， 我 们 
查询 了 用 户 名 和 java.library.path (在 程序 开头 和 结尾 处 不 同 寻常 的 注释 将 在 稍 后 进行 解释 。) 
2.7.1 编译 和 运行 

要 编译 、 运 行 这 个 程序 以 及 本 书 中 其 他 所 有 的 程序 ， 首 先 必须 要 有 一 个 Java 开 发 环境 。 目 
前 ， 有 相当 多 的 第 三 方 厂商 提供 开发 环境 ， 但 是 在 本 书 中 ， 我 假设 使 用 的 是 Sun 免 费 提 供 的 JDK 
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(Java Developer’s Kit, Java 开 发 人 员工 具 包 ) 开发 环境 。 若 使 用 其 他 的 开发 系统 ， 请 查找 该 系 
统 的 相应 文档 ， 以 便 决 定 怎样 编译 和 运行 程序 。 

请 登录 到 http:/java.sun.com 网 站 ， 那 里 会 有 相关 的 信息 和 链接 ， 引 导读 者 下 载 和 安装 与 自己 
的 机 器 平台 相 兼 容 的 JDK。 

安装 好 JDK 后 ， 还 需要 设 定好 路 径 信息 ， 以 确保 计算 机 能 找到 javac 和 java 这 两 个 文件 。 然 
后 请 下 载 并 解压 本 书 提供 的 源 代码 (从 wwwMindView net 处 可 以 获得 ) ， 它 会 为 书 中 每 一 章 自动 
创建 一 个 子 目录 。 请 转 到 object 子 目录 下 ， 并 键 和 人: 

javac HelloDate.java 

正常 情况 下 ， 这 行 命令 不 会 产生 任何 响应 。 如 果 有 任何 错误 消息 返回 给 你 ， 就 说 明 还 没 能 
正确 安装 好 JDK， 需 进一步 检查 并 找 出 问题 所 在 。 

如 果 没 有 返回 任何 回应 消息 ， 在 命令 提示 符 下 键入 : 

java HelloDate 

接着 , 便 可 看 到 程序 中 的 消息 和 当天 日 期 被 输出 。 

这 个 过 程 也 是 本 书 中 每 一 个 程序 的 编译 和 运行 过 程 。 然 而 ， 读 者 还 会 看 到 在 本 书 所 附 源 代 
码 中 ， 每 一 章 都 有 一 名 为 build.xml 的 文件 ， 访 文件 提供 一 个 “ant” 命 令 ， 用 于 自动 构建 该 章 的 
所 有 文件 。Build 文 件 和 Ant (以 及 在 哪里 下 载 ) 在 http://MindView.neVBooks/BetterJava 所 提供 的 
补充 材料 中 进行 了 完备 而 详细 的 讨论 。 一 旦 安装 好 Ant (可 从 http://jakarta.apache.org/ant 下 载 )， 
便 可 直接 在 命令 行 提示 符 下 键 人 ant 来 编译 和 运行 每 一 章 的 程序 了 。 如 果 尚 未 安装 Ant， 只 要 手 
工 键入 javac 和 java 命 令 即 可 安装 。 


2.8 注释 和 嵌入 式 文档 


Java 里 有 两 种 注释 风格 。 一 种 是 传统 的 C 语 言 风格 的 注释 一 -C++ 也 继承 了 这 种 风格 。 此 种 
注释 以 “/*” 开 始 ， 随 后 是 注释 内 容 ， 并 可 跨越 多 行 ， 最 后 以 “*/” 结 束 。 注 意 ,许多 程序 员 在 
连续 的 注释 内 容 的 每 一 行 都 以 一 个 “*” 开 头 ， 所 以 经 常 看 到 像 下 面 的 写法 : 


/* This is a comment 
* that continues 

* across lines 

*p 


但 请 记 住 ， 进 行 编译 时 ，/* 和 #/ 之 间 的 所 有 东西 都 会 被 忽略 ， 所 以 上 述 注释 与 下 面 这 段 注释 
并 没有 什么 两 样 : 


/* This is a comment that 
continues across lines */ 


第 二 种 风格 的 注释 也 源 于 C++。 这 种 注释 是 “单行 注释 "， 以 一 个 “/” 起 头 ， 直 到 句 末 。 
这 种 风格 的 注释 因为 书写 容易 ， 所 以 更 方便 、 更 常用 。 你 无 需 在 键盘 上 寻找 “/"， 再 寻找 “*” 
(而 只 需 按 两 次 同样 的 键 )， 而 且 不 必 考 虑 结束 注释 。 下 面 是 这 类 注释 的 例子 : 


// This is a one-line comment 


2.8.1 注释 文档 


代码 文档 撰写 的 最 大 问题 ， 大 概 就 是 对 文档 的 维护 了 。 如 果 文 档 与 代码 是 分 离 的 ， 那 么 在 


日 IBM 的 jikes 编 译 器 也 是 一 种 常用 的 编译 器 ， 它 比 Sun 的 javas 快 得 多 (尽管 你 可 以 用 Ant 来 构建 一 组 文件 ， 但 是 
这 不 会 有 太 大 的 差异 )。 另 外 还 有 很 多 创建 Java 编 译 器 、 运 行 时 环境 和 类 库 的 开源 项 目 。 
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每 次 修改 代码 时 ， 都 需要 修改 相应 的 文档 ， 这 会 成 为 一 件 相当 乏味 的 事情 。 解 决 的 方法 似乎 很 
简单 : 将 代码 同文 档 “ 链 接 ”起 来 。 为 达到 这 个 目的 ， 最 简单 的 方法 是 将 所 有 东西 都 放 在 同一 
个 文件 内 。 然 而 ， 为 实现 这 一 目的 ， 还 必须 使 用 一 种 特殊 的 注释 语法 来 标记 文档 ， 此 外 还 需 一 
个 工具 ， 用 于 提取 那些 注释 ， 并 将 其 转换 成 有 用 的 形式 。 这 正 是 Java 所 做 的 。 

javadoc 便 是 用 于 提取 注释 的 工具 , 它 是 JDK 安 装 的 一 部 分 。 它 采用 了 Java 编 译 器 的 某 些 技术 ， 
查找 程序 内 的 特殊 注释 标签 。 它 不 仅 解析 由 这 些 标签 标记 的 信息 ， 也 将 毗邻 注释 的 类 名 或 方法 
名 抽取 出 来 。 如 此 ， 我 们 就 可 以 用 最 少 的 工作 量 ， 生 成 相当 好 的 程序 文档 。 

javadoc 输 出 的 是 一 个 HTML 文 件 ， 可 以 用 Web 浏 览 器 查看 。 这 样 ， 该 工具 就 使 得 我 们 只 需 创 
建 和 维护 单一 的 源 文件 ， 并 能 自动 生成 有 用 的 文档 。 有 了 javadoc， 就 有 了 创建 文档 的 简明 直观 
的 标准 ， 我 们 可 以 期 望 、 甚 至 要 求 所 有 的 Java 类 库 都 提供 相关 的 文档 。 

此 外 ， 如 果 想 对 javadoc 处 理 过 的 信息 执行 特殊 的 操作 〈 例 如 ， 产 生 不 同 格式 的 输出 )， 那 么 
可 以 通过 编写 你 自己 的 被 称 为 “doclets” 的 javadoc 处 理 器 来 实现 。 关 于 doclets 在 http//MindView.net 
/Books/Better Java 所 提供 的 补充 材料 中 进行 了 介绍 。 4 

下 面 仅 对 基本 的 javadoc 进 行 简单 介绍 和 概述 。 全 面 翔 实 的 描述 可 从 java.sun.com 提 供 的 、 可 
下 载 的 JDK 文 档 中 找到 。( 注 意 此 文档 并 没有 与 JDK 一 块 打包 , 需 单独 下 载 。) 解压 缩 该 文档 之 后 ， 
查阅 “tooldocs” 子 目录 (或 点 击 “tooldocs” 链 接 )。 
2.8.2 语法 

所 有 javadoc 命 令 都 只 能 在 “/**” 注 释 中 出 现 ， 和 通常 一 样 ， 注 释 结束 于 “*/”"。 使 用 javadoc 
的 方式 主要 有 两 种 :做 入 HTML， 或 使 用 “文档 标签 "。 独 立 文档 标签 是 一 些 以 “@” 字 符 开头 
的 命令 ， 且 要 置 于 注释 行 的 最 前 面 (但 是 不 算 前 导 “*” 之 后 的 最 前 面 )。 而 “行内 文档 标签 ” 则 
可 以 出 现在 javadoc 注 释 中 的 任何 地 方 ， 它 们 也 是 以 “@” 字 符 开头 ， 但 要 括 在 花 括 号 内 。 

共有 三 种 类 型 的 注释 文档 ， 分 别 对 应 于 注释 位 置 后 面 的 三 种 元 素 : 类 、 域 和 方法 。 也 就 是 
说 ， 类 注释 正好 位 于 类 定义 之 前 ， 域 注释 正好 位 于 域 定义 之 前 ， ERUEN TAME 
义 的 前 面 。 如 下 面 这 个 简单 的 例子 所 示 : 


/1/: object/Documentation1.java 
/** A class comment */ 
public class Documentationl { 
/** A field comment */ & 
public int i; 
/** A method comment */ 
public void fO {} 
YM 
注意 ，javadoc 只 能 为 public (公共 ) 和 protected ( 受 保护 ) 成 员 进行 文档 注释 。private 
(私有 ) 和 包 内 可 访问 成 员 (参阅 第 5 章 ) 的 注释 会 被 忽略 掉 ， 所 以 输出 结果 中 看 不 到 它们 (不 
过 可 以 用 -private 进 行 标记 ， 以 便 把 private 成 员 的 注释 也 包括 在 内 ) 。 这 样 做 是 有 道理 的 ， 因 为 
只 有 public 和 protected 成 员 才 能 在 文件 之 外 被 使 用 ， 这 是 客户 端 程序 员 所 期 望 的 。 
上 述 代 码 的 输出 结果 是 一 个 HTML 文 件 ， 它 与 其 他 Java 文 档 具有 相同 的 标准 格式 。 因此， 用 
户 会 非常 熟悉 这 种 格式 ， 从 而 方便 地 导航 到 用 户 自己 设计 的 类 。 输 入 上 述 代 码 ， 然 后 通过 
javadoc 处 理 产生 HTML 文 件 ， 最 后 通过 浏览 器 观看 生成 的 结果 ， 这 样 做 是 非常 值得 的 。 
2.8.3 HRASCHTML 
javadoc 通 过 生成 的 HTML 文 档 传送 HTML 命 令 ， 这 使 你 能 够 充分 利用 HTML。 当然 ， 其 主要 
目的 还 是 为 了 对 代码 进行 格式 化 ， 例 如 : 


//: object/Documentation2.java 
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yee 
* <pre> 
* System.out.printin(new Date()); 


也 可 以 像 在 其 他 Web 文 档 中 那样 运用 HTML， 对 普通 文本 按照 你 自己 所 描述 的 进行 格式 化 : 


//: object/Documentation3.java 
ye 

* You can <em>even</em> insert a list: 
* <ol> 

* <li> Item one 

* <li> Item two 

* <li> Item three 
* </ol> 


注意 ， 在 文档 注释 中 ， 位 于 每 一 行 开头 的 星 号 和 前 导 空 格 都 会 被 javadoc 丢 弃 。javadoc 会 对 
所 有 内 容重 新 格式 化 ， 使 其 与 标准 的 文档 外 观 一 致 。 不 要 在 嵌入 式 HTML 中 使 用 标题 标签 ， 例 
如 <hl> 或 <hr>， 因 为 javadoc 会 插入 自 己 的 标题 ， 而 你 的 标题 可 能 同 它们 发 生 冲突 。 

所 有 类 型 的 注释 文档 -一 类 、 域 和 方法 一 一 都 支持 嵌 人 式 HTML。 
2.8.4 一 些 标签 示例 

这 里 将 介绍 一 些 可 用 于 代码 文档 的 javadoc 标 签 。 在 使 用 javadoc 处 理 重要 事情 之 前 ， 应 该 先 
到 JDK 文 档 那里 查阅 javadoc 参 考 ， 以 学 习 javadoc 的 各 种 不 同 的 使 用 方法 。 

1. @see: 引用 其 他 类 

@see 标 签 允 许 用 户 引用 其 他 类 的 文档 。javadoc 会 在 其 生成 的 HTML 文 件 中 ， 通 过 @see 标 签 
链接 到 其 他 文档 。 格 式 如 下 : 


@see classname 
@see fully-qualified-classname 
@see fully-qualified-classname#method-name 


上 述 每 种 格式 都 会 在 生成 的 文档 中 加 入 一 个 具有 超 链接 的 “See Also” (SL) 条 目 。 但 是 
javadoc 不 会 检查 你 所 提供 的 超 链接 是 否 有 效 。 

2. {@link package.class#member label} 

该 标签 与 @see 极 其 相似 ， 只 是 它 用 于 行内 ， 并 且 是 用 “label” 作 为 超 链接 文本 而 不 用 “See 
Also”, 

3. {@docRoot} 

该 标签 产生 到 文档 根 目录 的 相对 路 径 ， 用 于 文档 树 页 面 的 显 式 超 链接 。 

4. {@inheritDoc} i 

该 标签 从 当前 这 个 类 的 最 直接 的 基 类 中 继承 相关 文档 到 当前 的 文档 注释 中 。 

5. @version 

该 标签 的 格式 如 下 : 

@version version-information 
其 中 ,“version-information” 可 以 是 任何 你 认为 适合 包含 在 版 本 说 明 中 的 重要 信息 。 如 果 
javadoc 命 令 行 使 用 了 “-version” 标 记 ， 那 么 就 从 生成 的 HTML 文 档 中 特别 提取 出 版 本 信息 。 

6. @author 

该 标签 的 格式 如 下 : 


eee ee OS eee eS 
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@author author-information 

其 中 ，author-information 一 看 便 知 是 你 的 姓名 ， 但 是 也 可 以 包括 电子 邮件 地 址 或 者 其 他 任 
何 适宜 的 信息 。 如 果 javadoc 命 令 行 使 用 了 -author 标 记 ， 那 么 就 从 生成 的 HTML 文 档 中 特别 提取 
作者 信息 。 

可 以 使 用 多 个 标签 ， 以 便 列 出 所 有 作者 ， 但 是 它们 必须 连续 放置 。 全 部 作者 信息 会 合并 到 
同一 段落 ， 置 于 生成 的 HTML 中 。 

7. @since 

该 标签 允许 你 指定 程序 代码 最 早 使 用 的 版 本 ， 可 以 在 HTML Java 文 档 中 看 到 它 被 用 来 指定 
所 用 的 JDK 版 本 的 情况 。 

8. @param 

该 标签 用 于 方法 文档 中 ， 形 式 如 下 : 

@param parameter-name description 
其 中 ，parameter-name 是 方法 的 参数 列表 中 的 标识 符 ，description 是 可 延续 数 行 的 文本 ， 终 止 于 
新 的 文档 标签 出 现 之 前 。 可 以 使 用 任意 多 个 这 种 标签 ， 大 约 每 个 参数 都 有 一 个 这 样 的 标签 。 

9. @return 

该 标签 用 于 方法 文档 ， 格 式 如 下 : 

@return description 
其 中 “deseription” 用 来 描述 返回 值 的 含义 ， 可 以 延续 数 行 。 Me 

10. @throws 

“异常 ”将 在 第 9 章 论述 。 简 言 之 ， 它 们 是 由 于 某 个 方法 调用 失败 而 “ 抛 出 ”的 对 象 。 尽 管 
在 调用 一 个 方法 时 ， 只 出 现 一 个 异常 对 象 ， 但 是 某 个 特殊 方法 可 能 会 产生 任意 多 个 不 同类 型 的 
异常 ， 所 有 这 些 异常 都 需要 进行 说 明 。 所 以 ， 异 常 标签 的 格式 如 下 ;: 

@throws fully-qualified-class-name description 
其 中 fully-qualified-class-name 给 出 一 个 异常 类 的 无 歧义 的 名 字 ， 而 该 异常 类 在 别处 定义 。 
description (同样 可 以 延续 数 行 ) 告诉 你 为 什么 此 特殊 类 型 的 异常 会 在 方法 调用 中 出 现 。 

11, @deprecated 

该 标签 用 于 指出 一 些 旧 特 性 已 由 改进 的 新 特性 所 取代 ， 建 议 用 户 不 要 再 使 用 这 些 旧 特性 ， 
因为 在 不 久 的 将 来 它们 很 可 能 会 被 刷 除 。 如 果 使 用 一 个 标记 为 @deprecated 的 方法 ， 则 会 引起 编 
译 器 发 布 警告 。 

在 Java SE5 中 ，Javadoc 标 签 @deprecated 已 经 被 @Deprecated 注 解 所 替代 (我们 将 在 第 20 章 中 
学 习 相关 的 知识 。) 
2.8.5 文档 示例 

下 面 再 回 到 第 一 个 Java 程 序 ， 但 是 这 次 加 上 了 文档 注释 : 














//: object/HelloDate. java 
import java.util.*; 


/** The first Thinking in Java example program. 
* Displays a string and today's date. 
* @author Bruce Eckel 
* @author www.MindView.net 
* @version 4.0 
" 
public class HelloDate { 
/** Entrv point to class & application. 
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* @param args array of string arguments 
* @throws exceptions No exceptions thrown 
“ 

public static void main(String[] args) { 


System.out.printIn("Hello, it's: "); 
System.out.printin(new Date()); 


} 
} /* Output: (55% match) 
Hello, it’s: 
Wed Oct 05 14:39:36 MDT 2005 
Whim 


第 一 行 采用 我 自己 独特 的 方法 ， 用 一 个 “:” 作 为 特殊 记号 说 明 这 是 包含 源 文 件 名 的 注释 行 。 
该 行 包含 文件 的 路 径 信息 (此 时 ，object 代 表 本 章 )， 随 后 是 文件 名 。 最 后 一 行 也 是 一 行 注释 ， 
这 个 “W:~” 标 志 源 代码 清单 的 结束 。 自 此 ， 在 通过 编译 器 和 执行 检查 后 ， 文档 就 可 以 自动 更 新 
成 本 书 的 文本 。 

/*Output 标 签 表示 输出 的 开始 部 分 将 由 这 个 文件 生成 ， 通 过 这 种 形式 ， 它 会 被 自动 地 测试 
以 验证 其 准确 性 。 在 本 例 中 ，(55% match) 在 向 测试 系统 说 明 程序 的 每 一 次 运行 和 下 一 次 运行 
的 输出 存在 着 很 大 的 差异 ， 因 此 它们 与 这 里 列 出 的 输出 预期 只 有 55% 的 相关 性 。 本 书 中 能 够 产 
生 输 出 的 大 部 分 示例 都 包含 这 种 注释 方式 的 输出 ， 因 此 你 可 以 查看 它们 的 运行 输出 ， 并 知晓 其 
正确 性 。 


2.9 编码 风格 


在 “Java 编 程 语言 编码 约定 ” 中， 代码 风格 是 这 样 规定 的 : 类 名 的 首 字母 要 大 写 ， 如 果 类 
名 由 儿 个 单词 构成 ， 那 么 把 它们 并 在 一 起 (也 就 是 说 ， 不 要 用 下 划 线 来 分 隔 名 字 )， 其 中 每 个 内 
部 单词 的 首 字母 都 采用 大 写 形式 。 例 如 : 


class AllTheColorsOfTheRainbow { // ... 


这 种 风格 有 时 称 作 “ 驼 峰 风格 "。 儿 乎 其 他 所 有 内 容 一 方法 、 字 段 (成 员 变量 ) 以 及 对 象 
引用 名 称 等 ， 公 认 的 风格 与 类 的 风格 一 样 ， 只 是 标识 符 的 第 一 个 字母 采用 小 写 。 例 如 ， 


class ALLTheColorsOfTheRainbow { 
int anIntegerRepresentingCotors; 
void changeTheHueOfTheColor(int newHue) { 
ID See 
} 
Moe 


} 


当然 ， 用 户 还 必须 键入 所 有 这 些 长 名 字 ， 并 且 不 能 输 错 ， 因 此 ， 一 定 要 格外 仔细 。 
Sun 程 序 库 中 的 Java 代 码 也 采用 本 书 摆 放 开 、 闭 花 括号 的 方式 。 


2.10 总 结 


通过 本 章 的 学 习 ， 大 家 已 接触 相当 多 的 关于 如 何 编写 一 个 简单 程序 的 Java 编 程 知识 。 此 外 ， 
对 Java 语 言及 它 的 一 些 基 本 思想 也 有 了 一 个 总 体 认识 。 然 而 到 目前 为 止 ， 所 有 示例 都 是 “这 样 
做 ， 再 那样 做 ， 接 着 再 做 另 一 些 事情 ”这 种 形式 。 如 果 想 让 程序 做 出 选择 ， 例 如 ,“ 假 如 所 做 的 
结果 是 红色 ， 就 那样 做 ， 否 则 ， 就 做 另 一 些 事情 "。 这 将 又 怎样 进行 呢 ? Java 对 此 种 基本 编程 行 
为 所 提供 的 支持 ， 将 会 在 下 一 章 讲述 。 
© 网 址 是 :httpV/iavasun conydocsieodeconvindex html。 为 了 节省 本 书 和 课 生 演示 的 篇 旺 ， 没 有 遵循 约 定 中 的 全 
部 条 款 ， 但 是 你 将 会 看 到 我 在 这 里 所 使 用 的 风格 尽 可 能 地 与 Java 标 准 相 匹配 。 
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2.11 练习 


所 选 习题 的 答案 都 可 以 在 名 为 “The Thinking in Java Annotated Solution Guide” 的 电子 文档 
中 找到 ， 读 者 可 以 从 www.MindView.net 处 购买 此 文档 。 

练习 1: (2) 创建 一 个 类 ， 它 包含 一 个 int 域 和 一 个 char 域 ， 它 们 都 没有 被 初始 化 ， 将 它们 的 
值 打印 出 来 ， 以 验证 Java 执 行 了 默认 初始 化 。 

练习 2: (1) 参照 本 章 的 HelloDatejava 这 个 例子 ， 创 建 一 个 “Hello，World” 程 序 ， 该 程序 
只 要 输出 这 句 话 即 可 。 你 所 编写 的 类 里 只 需 一 个 方法 〈 即 “main” 方 法 ， 在 程序 启动 时 被 执行 ) 。 
记 住 要 把 它 设 为 static 形 式 ， 并 指定 参数 列表 一 即使 根本 不 会 用 到 这 个 列表 。 用 javac 进 行 编译 ， 
再 用 java 运 行 它 。 如 果 你 使 用 的 是 不 同 于 JDK 的 开发 环境 ， 请 了 解 如 何在 你 的 环境 中 进行 编译 和 
运行 。 

练习 3; (1) 找 出 含有 ATypeName 的 代码 段 ， 将 其 改写 成 完整 的 程序 ， 然 后 编译 、 运 行 。 

练习 4: (1) 将 DataOnly 代 码 段 改写 成 一 个 程序 ， 然 后 编译 、 运 行 。 

练习 5: (1) 修改 前 一 个 练习 ， 将 DataOnly 中 的 数据 在 main0 方 法 中 赋值 并 打印 出 来 。 

练习 6: (2) 编写 一 个 程序 ， 让 它 含有 本 章 所 定义 的 storage0 方 法 的 代码 段 ， 并 调用 之 。 

练习 7: (1) 将 Incrementable 的 代码 段 改 写成 一 个 完整 的 可 运行 程序 。 

练习 8: (3) 编写 一 个 程序 ， 展 示 无 论 你 创建 了 某 个 特定 类 的 多 少 个 对 象 ， 这 个 类 中 的 某 个 
特定 的 static 域 只 有 一 个 实例 。 

练习 9: (2) 编写 一 个 程序 ， 展 示 自动 包装 功能 对 所 有 的 基本 类 型 和 包装 器 类 型 都 起 作用 。 

练习 10，(2) 编写 一 个 程序 ， 打 印 出 从 命令 行 获得 的 三 个 参数 。 为 此 ， 需 要 确定 命令 行 数组 
中 String 的 下 标 。 

练习 11: (1) 将 AITheColorsOfTheRainbow 这 个 示例 改写 成 一 个 程序 ， 然 后 编译 、 运 行 。 

练习 12: (2) 找 出 HelloDatejava 的 第 二 版 本 ， 也 就 是 那个 简单 注释 文档 的 示例 。 对 该 文件 
执行 javadoc， 然 后 通过 Web 浏 览 器 观看 运行 结果 。 

练习 13， (1) 通过 Javadoc 运 行 Documentationljava，Documentation2.java 和 Documen- 
tation3java， 然 后 通过 Web 浏 览 器 验证 所 产生 的 文档 。 

练习 14: (1) 在 前 一 个 练习 的 文档 中 加 入 各 项 的 HTML 列 表 。 

练习 15: (1) 使 用 练习 2 的 程序 ， 加 入 注释 文档 。 用 javadoc 提 取 此 注释 文档 ， 并 产生 一 个 
HTML 文 件 ， 然 后 通过 Web 浏 览 器 查看 结果 。 

练习 16: (1) 找到 第 5 章 中 的 Overloading.java 示 例 ， 并 为 它 加 入 javadoc 文 档 。 然后 用 
javadoc 提 取 此 注释 文档 ， 并 产生 一 个 HTML 文 件 ， 最 后 ， 通 过 Web 浏 览 器 查看 结果 。 





[39] 
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第 3 章 操作 符 


在 最 底层 ，Java 中 的 数据 是 通过 使 用 操作 符 来 操 做 的 。 

Java 是 建立 在 C++ 基础 之 上 的 ， 所 以 C 和 C++ 程序 员 应 该 非常 熟悉 Java 的 大 多 数 操作 符 。 当 
然 ，Java 也 做 了 一 些 改进 与 简化 。 

如 果 读 者 熟悉 C 或 C++ 的 语法 ， 那 么 只 需 快速 浏览 本 章 和 下 一 章 ， 看 看 Java 与 这 些 语言 之 间 
的 差异 。 但 是 ， 如 果 读 者 觉得 很 难 理解 这 两 章 的 内 容 ， 那 就 请 您 先 阅读 可 以 从 www.MindViewnet 
上 免费 下 载 的 多 媒体 课程 《Thinking in C》， 其 中 包括 精心 设计 的 有 声 讲解 、 幻 灯 片 、 练 习 以 及 解 
答 ， 这 些 能 带领 读者 快速 掌握 学 习 Java 所 必需 的 基础 知识 。 


3.1 更 简单 的 打印 语句 
在 前 一 章 中 ， 我 们 介绍 了 Java 的 打印 语句 。 


System.out,printtn("Rather a lot to type"); 


你 可 以 看 到 ， 这 条 语句 不 仅 涉及 许多 类 型 (因此 有 许多 多 余 的 连接 )， 而 且 它 读 起 来 也 颇 为 
费劲 。 在 Java 之 前 和 之 后 出 现 的 大 多 数 语言 都 已 经 采取 了 一 种 简单 得 多 的 方式 来 提供 这 种 常用 
语句 。 

在 第 6 章 中 将 介绍 静态 导入 (static import) 这 个 在 Java SE5 中 新 增加 的 概念 ， 并 将 创建 一 个 
小 类 库 来 简化 打印 语句 的 编写 。 但 是 ， 在 开始 使 用 这 个 类 库 之 前 ， 不 必 先 去 了 解 其 中 的 细节 。 
通过 使 用 这 个 新 类 库 ， 可 以 把 上 一 章 中 的 程序 改写 如 下 : 


1/: operators/HelloDate. java 
import java.util.*; 
import static net.mindview.util.Print.*; 


public class HelloDate { 
public static void main(String{] args) { 
print("Hello, it's: "); 
print (new Date()); 


} 
} /* Output: (55% match) 
Hello, it's: 
Wed Oct 05 14:39:05 MDT 2005 
Whim 


改写 后 的 程序 清爽 了 许多 。 请 注意 ， 我 们 在 第 二 个 import 语 句 中 插入 了 static 关 键 字 。 

要 想 使 用 这 个 类 库 ， 必 须 从 www.MindView.net 或 其 镜像 之 一 下 载 本 书 的 代码 包 ， 然 后 解压 
代码 目录 树 ， 并 在 你 的 计算 机 的 CLASSPATH 环 境 变量 中 添加 该 代码 目录 树 的 根 目录 。( 你 最 终 
会 获得 有 关 类 路 径 的 完整 介绍 ， 但 是 你 也 应 该 尽早 习惯 它 带 来 的 麻烦 。 唉 ， 它 是 你 在 使 用 Java 
时 最 常见 的 问题 之 一 。) 

尽管 使 用 net.mindview.util.Print 可 以 很 好 地 简化 大 多 数 的 代码 ， 但 是 它 并 非 在 任何 场合 
都 显得 很 恰当 。 如 果 在 代码 中 只 有 少量 的 打印 语句 ， 我 还 是 先 用 import 然 后 编写 完整 的 
System.out.printinQ), 

练习 1: (DD) 使 用 “简短 的 ”和 正常 的 打印 语句 来 编写 一 个 程序 。 
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3.2 使 用 Java 操 作 符 


操作 符 接受 一 个 或 多 个 参数 ， 并 生成 一 个 新 值 。 参 数 的 形式 与 普通 的 方法 调用 不 同 ， 但 效 
果 是 相同 的 。 加 号 和 一 元 的 正 号 (+)、 减 号 和 一 元 的 负 号 (一 )、 乘 号 (*)、 除 号 (/) 以 及 赋值 
号 (=) 的 用 法 与 其 他 编程 语言 类 似 。 

操作 符 作 用 于 操作 数 ， 生 成 一 个 新 值 。 另 外 ， 有 些 操作 符 可 能 会 改变 操作 数 自身 的 值 ， 这 
被 称 为 “副作用 ”"。 那 些 能 改变 其 操作 数 的 操作 符 ， 最 普遍 的 用 途 就 是 用 来 产生 副作用 ， 但 要 记 
住 ,使 用 此 类 操作 符 生成 的 值 ， 与 使 用 没有 副作用 的 操作 符 生成 的 值 ， 没 有 什么 区 别 。 

几乎 所 有 的 操作 符 都 只 能 操作 “基本 类 型 "。 例 外 的 操作 符 是 “=”"、“==” 和 “!=”， 这些 
操作 符 能 操作 所 有 的 对 象 (这 也 是 对 象 易 令 人 糊涂 的 地 方 )。 除 此 以 外 ，String 类 支持 “+” 和 


“=”, 


3.3 优先 级 


当 一 个 表达 式 中 存在 多 个 操作 符 时 ， 操 作 符 的 优先 级 就 决定 了 各 部 分 的 计算 顺序 。Java 对 
计算 顺序 做 了 特别 的 规定 。 其 中 ， 最 简单 的 规则 就 是 先 乘除 后 加 减 。 程 序 员 经 常会 忘记 其 他 优 
先 级 规则 ， 所 以 应 该 用 括号 明确 规定 计算 顺序 。 例 如 ， 以 下 语句 中 的 (0 和 (2): 


//: operators/Precedence. java 


public class Precedence { 
public static void wmetni(seringl) args) { 
intx=1, y= 2,23; 
inta=x+y- 2/242; 1 (1) 
int b= x + (y - 2)/Q + 2); MQ) 
System.out.printin("a-= "+ a+" b=" +b); 


} 
} /* Output: 


a=5b=1 
SHI i~ 


这 两 个 语句 看 起 来 大 体 相 同 ， 但 是 从 输出 就 可 以 看 出 它们 具有 过 然 不 同 的 含义 ， 而 这 正 是 
使 用 括号 的 结果 。 

请 注意 ，System.outprintin0 语 句 中 包含 “+” 操 作 符 。 在 这 种 上 下 文 环 境 中 ,“+” 意 味 着 
“字符 串 连接 ”， 并 且 如 果 必 要 ， 它 还 要 执行 “字符 串 转换 "。 当 编译 器 观察 到 一 个 String 后 面 紧 
跟 一 个 “+”， 而 这 个 “+” 的 后 面 又 紧 跟 一 个 非 String 类 型 的 元 素 时 ， 就 会 尝试 着 将 这 个 非 String 
类 型 的 元 素 转换 为 String。 正 如 在 输出 中 所 看 到 的 ， 它 成 功 地 将 a 和 b 从 int 转 换 为 String 了 。 


3.4 赋值 


赋值 使 用 操作 符 “=”。 它 的 意思 是 “ 取 右边 的 值 〈 即 右 值 )， 把 它 复制 给 左边 〈 即 左 值 )”。 
右 值 可 以 是 任何 常数 、 变 量 或 者 表达 式 (只 要 它 能 生成 一 个 值 就 行 )。 但 左 值 必须 是 一 个 明确 的 、 
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已 命名 的 变量 。 也 就 是 说 ， 必 须 有 一 个 物理 空间 可 以 存储 等 号 右边 的 值 。 举 例 来 说 ， 可 将 一 个 
常数 赋 给 一 个 变量 : 

a=4; 

但 是 不 能 把 任何 东西 赋 给 一 个 常数 ， 常 数 不 能 作为 左 值 ( 比 如 不 能 说 4=a，)。 

对 基本 数据 类 型 的 赋值 是 很 简单 的 。 基 本 类 型 存储 了 实际 的 数值 ， 而 并 非 指向 一 个 对 象 的 
引用 ， 所 以 在 为 其 赋值 的 时 候 ， 是 直接 将 一 个 地 方 的 内 容 复制 到 了 另 一 个 地 方 。 例 如 ， 对 基本 
数据 类 型 使 用 a=b, 那么 b 的 内 容 就 复制 给 a。 若 接着 又 修改 了 a, 而 b 根 本 不 会 受 这 种 修改 的 影响 。 
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作为 程序 员 ， 这 正 是 大 多 数 情况 下 我 们 所 期 望 的 。 

但 是 在 为 对 象 “赋值 ”的 时 候 ， 情 况 却 发 生 了 变化 。 对 一 个 对 象 进行 操作 时 ， 我 们 真正 操 
作 的 是 对 对 象 的 引用 。 所 以 倘若 “将 一 个 对 象 赋值 给 另 一 个 对 象 "， 实 际 是 将 “引用 ”从 一 个 地 
方 复制 到 另 一 个 地 方 。 这 意味 着 假若 对 对 象 使 用 c=d， 那 么 c 和 d 都 指向 原本 只 有 d 指 向 的 那个 对 
Re PHT AFH AAR MAA 


//: operators/Assignment.java 
// Assignment with objects is a bit tricky. 
import static net.mindview.util.Print.*; 


class Tank { 
int level; 
} 


public class Assignment { 
public static void main(String[] args) { 

Tank tl = new Tank(); 

Tank t2 = new Tank(); 

t1.level = 9; 

t2.level = 47; 

print("1: ti.level: " + ti.level + 
*, t2.level: "+ t2. level); 

tl = t2; 

print("2: tl.level: " + ti.level + 
", t2.level: " + t2.level); 

tl.level = 27; 

print("3: tl.level: " + ti.level + 
", t2. level: " + t2. level); 


} 
} /* Output: 
1: tl.level: 9, t2.level: 47 
2: tl.level: 47, t2.level: 47 
3: tl.tevel: 27, t2.level: 27 
AAA 


Tank 类 非常 简单 ， 它 的 两 个 实例 (tL 和 t2) 是 在 main0) 里 创建 的 。 对 每 个 Tank 类 对 象 的 
level 域 都 冉 予 了 一 个 不 同 的 值 ， 然 后 ， 将 t2 赋 给 tL， 接 着 又 修改 了 tL。 在 许多 编程 语言 中 ， 我 们 
可 能 会 期 望 tL 和 2 总 是 相互 独立 的 。 但 由 于 赋值 操作 的 是 一 个 对 象 的 引用 ， 所 以 修改 的 同时 也 
改变 了 t2! 这 是 由 于 tL 和 2 包 含 的 是 相同 的 引用 ， 它 们 指向 相同 的 对 象 。( 原 本 (包含 的 对 对 象 
的 引用 ， 是 指向 一 个 值 为 9 的 对 象 。 在 对 和 赋值 的 时 候 ， 这 个 引用 被 覆盖 ， 也 就 是 丢失 了 ， 而 那 
个 不 再 被 引用 的 对 象 会 由 “垃圾 回收 器 ”自动 清理 。) 

这 种 特殊 的 现象 通常 称 作 “ 别 名 现象 "， 是 Java 操 作对 象 的 一 种 基本 方式 。 在 这 个 例子 中 ， 
如 果 想 避免 别名 问题 应 该 怎么 办 呢 ? 可 以 这 样 写 : 

tl.level = t2.Leveli 

PEETLA RR Pt Re A, TAAL AE BIA, (PRR RRS RI 
到 ， 直 接 操作 对 象 内 的 域 容易 导致 混乱 ， 并 且 ， 违 背 了 良好 的 面向 对 象 程序 设计 的 原则 。 这 可 
不 是 一 个 小 问题 ， 所 以 从 现在 开始 大 家 就 应 该 留意 ， 为 对 象 赋值 可 能 会 产生 意 想不到 的 结果 。 

练习 2: (1) 创建 一 个 包含 一 个 float 域 的 类 ， 并 用 这 个 类 来 展示 别名 机 制 。 
3.4.1 方法 调用 中 的 别名 问题 

将 一 个 对 象 传递 给 方法 时 ， 也 会 产生 别名 问题 : 


7/: operators/PassObject.java 

47/ Passing objects to methods may not. be 
// what you're used to. 

import static net.mindview.utit.Print.*; 
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class Letter { 
char c; 
} 
public class PassObject { 
static void f(Letter y) { 
y.c = '2'3 


public static void main(String[] args) { 





print("2: x.c: " + x.c); 


} 
} /* Output: 
1: x.c: a 
A 3 
i~ 


在 许多 编程 语言 中 ， 方 法 f0 似 乎 要 在 它 的 作用 域内 复制 其 参数 Letter y 的 一 个 副本 ， 但 实际 
上 只 是 传递 了 一 个 引用 。 所 以 代码 行 

y.c = °2"; 

实际 改变 的 是 f0 之 外 的 对 象 。 

别名 引起 的 问题 及 其 解决 办 法 是 很 复杂 的 话题 ， 本 书 的 在 线 补充 材料 涵盖 了 此 话题 。 但 是 
你 现在 就 应 该 知道 它 的 存在 ， 并 在 使 用 中 注意 这 个 陷阱 。 

练习 3: (1) 创建 一 个 包含 一 个 float 域 的 类 ， 并 用 这 个 类 来 展示 方法 调用 时 的 别名 机 制 。 


3.5 算术 操作 符 


Java 的 基本 算术 操作 符 与 其 他 大 多 数 程序 设计 语言 是 相同 的 。 其 中 包括 加 号 (+)、 减 号 (一 )、 
BRE 〈/)、 乘 号 (*) 以 及 取 模 操作 符 (%， 它 从 整数 除法 中 产生 余数 )。 整 数 除法 会 直接 去 掉 结 
果 的 小 数位 ， 而 不 是 四 舍 五 入 地 贺 整 结果 。 

Java 也 使 用 一 种 来 自 C 和 C++ 的 简化 符号 同时 进行 运算 与 赋值 操作 。 这 用 操作 符 后 紧 跟 一 个 
等 号 来 表示 ， 它 对 于 Java 中 的 所 有 操作 符 都 适用 ， 只 要 其 有 实际 意义 就 行 。 例 如 ， 要 将 x 加 4， 
并 将 结果 赋 回 给 x， 可 以 这 么 写 : x+=4。 

下 面 这 个 例子 展示 了 各 种 算术 操作 符 的 用 法 : 


//: operators/MathOps. java 
// Demonstrates the mathematical operators. 
import java.util.*; 

import static net.mindview.util.Print.*; 


public class MathOps { 
public static void main(String{] args) { _ 

// Create a seeded random number generator: 
Random rand = new Random(47); 
int i, j,k; 
// Choose value from 1 to 100: 
j = rand.nextInt (160) + 1; 
print("j : " + j); 





j): 
k = ral tInt(100) + 1 
print("k : "+ k); 
i=j+k; 

print("j + k : * +i); 
i=j 

print "j +1): 








1=k1ji 
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print("k / j: "+0; 


pt + ids 





print Ck A T+: 

j %= 

ae “j = ko" +j); 

// Floating-point number tests: 

float u, v, w; // Applies to doubles, too 
v = rand.nextFloat(); 

print("v : " + v); 

w = rand.nextFloat(); 

print(" : 

u=v+ 
print(” 





print("v / w : " +u); 

// The following also works for’ char, 
// byte, short, int, long, and double: 
u +s v; 











print("u += v : * +u); 
u -= v; 
print("u -= +u); 
yty 
print("u *= "+ 0); 
u/ev; 


print("u /= v : "+ u); 


} 
/* Output: 





} 

j: 

k:t 

j 

j 

k 

k 

k 

j 

v 

W 

v : @.5843576 
v : 0.47753322 
v : 6.928358962 
v : 9.940527 
u : 10.471473 
u : 9.940527 
u 5.2778773 
u : 9.940527 





要 生成 数字 , 程序 首先 会 创建 一 个 Random 类 的 对 象 。 如 果 在 创建 过 程 中 没有 传递 任何 参数 ， 
那么 Java 就 会 将 当前 时 间作 为 随机 数 生成 器 的 种 子 ， 并 由 此 在 程序 每 一 次 执行 时 都 产生 不 同 的 
输出 。 但 是 ， 在 本 书 的 示例 中 ， 示 例 末尾 所 展示 的 输出 都 尽 可 能 一 致 ， 这 一 点 很 重要 ， 因 为 这 
样 就 使 得 这 些 输出 可 以 用 外 部 工具 来 验证 。 通 过 在 创建 Random 对 象 时 提供 种 子 (用 于 随机 数 生 
成 器 的 初始 化 值 ， 随 机 数 生成 器 对 于 特定 的 种 子 值 总 是 产生 相同 的 随机 数 序列 ) ， 就 可 以 在 每 一 
次 执行 程序 时 都 生成 相同 的 随机 数 ， 因 此 其 输出 是 可 验证 的 。 要 想 生成 更 多 的 各 种 不 同 的 输 
出 ， 可 以 随意 移 除 本 书 示例 中 的 种 子 。 


O 数字 47 在 我 加 盟 的 一 家 学 院 里 被 认为 是 “魔幻 数字 "， 至 今 仍 是 这 样 。 
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通过 Random 类 的 对 象 ， 程 序 可 生成 许多 不 同类 型 的 随机 数字 。 做 法 很 简单 ， 只 需 调用 方法 
nextInt0 和 nextFloat0 即 可 (也 可 以 调用 nextLong0 或 者 nextDouble0)。 传 递 给 nextInt0 的 参数 
设置 了 所 产生 的 随机 数 的 上 限 ， 而 其 下 限 为 0， 但 是 这 个 下 限 并 不 是 我 们 想 要 的 ， 因 为 它 会 产生 
除 0 的 可 能 性 ， 因 此 我 们 对 结果 做 了 加 1 操作 。 

练习 4: (2) 编写 一 个 计算 速度 的 程序 ， 它 所 使 用 的 距离 和 时 间 都 是 常量 。 
3.5.1 一 元 加 、 减 操作 符 

TORS (-) 和 一 元 加 号 (+) 与 二 元 减 号 和 加 号 都 使 用 相同 的 符号 。 根 据 表达 式 的 书写 
形式 ， 编 译 器 会 自动 判断 出 使 用 的 是 哪 一 种 。 例 如 语句 
的 含义 是 显然 的 。 编 译 器 能 正确 识别 下 述 语句 : 

X=a* -b 
但 读者 会 被 搞 糊 涂 ， 所 以 有 时 更 明确 地 写成 ， 

x=a* (-b); 

一 元 减 号 用 于 转变 数据 的 符号 ， 而 一 元 加 号 只 是 为 了 与 一 元 减 号 相对 应 ,但 是 它 唯一 的 作 
用 仅仅 是 将 较 小 类 型 的 操作 数 提升 为 int。 


3.6 自动 递增 和 递减 


和 C 类 似 ，Java 提 供 了 大 量 的 快捷 运算 。 这 些 快捷 运算 使 编码 更 方便 ， 同 时 也 使 得 代码 更 容 
易 阅 读 ， 但 是 有 时 可 能 使 代码 阅读 起 来 更 困难 。 

递增 和 递减 运算 是 两 种 相当 不 错 的 快捷 运算 〈( 常 称 为 “自动 递增 ”和 “自动 递减 ”运算 ) 。 
其 中 ， 递 减 操作 符 是 “--”， 意 为 “减少 一 个 单位 ”， 递增 操作 符 是 “++”， 意 为 “增加 一 个 单 
位 "。 举 个 例子 来 说 ， 假 设 a 是 一 个 int (整数 ) 值 ， 则 表达 式 ++a 就 等 价 于 (a = a + 1)。 递 增 和 
递减 操作 符 不 仅 改变 了 变量 ， 并 且 以 变量 的 值 作 为 生成 的 结果 。 

这 两 个 操作 符 各 有 两 种 使 用 方式 ， 通 常 称 为 “前 组 式 ” 和 “后 级 式 "。“ 前 组 递增 ”表示 
“++” 操 作 符 位 于 变量 或 表达 式 的 前 面 ， 而 “后 绥 递 增 ” 表 示 “++” 操 作 符 位 于 变量 或 表达 式 
的 后 面 。 类 似 地 ,“ 前 级 递 碱 ”意味 着 “--” 操 作 符 位 于 变量 或 表达 式 的 前 面 ， 而 “后 级 递 碱 ” 
意味 着 “--” 操 作 符 位 于 变量 或 表达 式 的 后 面 。 对 于 前 绿 递 增 和 前 绥 递 碱 (如 ++a 或 -a)， 会 先 
执行 运算 ， 再 生成 值 。 而 对 于 后 缘 递 增 和 后 丝 递 碱 (如 a++ 或 a--)， 会 先生 成 值 ， 再 执行 运算 。 
下 面 是 一 个 例子 : 


/1/: operators/AutoInc. java 
// Demonstrates the ++ and -- operators. 
import static net.mindview.util.Print.*; 


public class AutoInc { 
public static void main(String] args) { 
int 1 = 1; 











i): // Pre-increment 
i // Post-increment 


i // Pre-decrement 
i // Post-decrement 
print("i : 
} 
} /* Output: 
ETS 


+i 2 
it+ 2 





4 3# 
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从 中 可 以 看 到 ， 对 于 前 绥 形 式 ， 我 们 在 执行 完 运算 后 才 得 到 值 。 但 对 于 后 绥 形 式 ， 则 是 在 
运算 执行 之 前 就 得 到 值 。 它 们 是 除 那些 涉及 赋值 的 操作 符 以 外 ， 唯 一 具有 “副作用 ”的 操作 符 。 


也 就 是 说 ， 它 们 会 改变 操作 数 ， 而 不 仅仅 是 使 用 自己 的 值 。 

递增 操作 符 正 是 对 C++ 这 个 名 字 的 一 种 解释 ， 暗 示 着 “超越 C 一 步 "。 在 早期 的 一 次 有 关 Java 
HORE, Bill Joy (Java 创 始 人 之 一 ) 声称 “Java=C++--”(C 加 加 减 减 ) 意味 着 Java 已 去 除了 
C++ 中 一 些 很 困难 而 又 没 必要 的 东西 ， 成 为 了 一 种 更 精简 的 语言 。 正 如 大 家 会 在 这 本 书 中 学 到 
的 ，Java 的 许多 地 方 更 精简 了 ， 然 而 并 不 是 说 Java 在 其 他 方面 也 比 C++ 容易 很 多 。 


3.7 关系 操作 符 


关系 操作 符 生成 的 是 一 个 boolean (布尔 ) 结果 ， 它 们 计算 的 是 操作 数 的 值 之 间 的 关系 。 如 
果 关 系 是 真实 的 ， 关 系 表 达 式 会 生成 true ( 真 ) ， 如 果 关 系 不 真实 ， 则 生成 false ( 假 )。 关 系 操 
作 符 包括 小 于 (<), KF (>)、 小 于 或 等 于 (<=)、 大 于 或 等 于 (>=), SF (==) 以 及 不 等 于 
(!=)。 等 于 和 不 等 于 适用 于 所 有 的 基本 数据 类 型 ， 而 其 他 比较 符 不 适用 于 boolean 类 型 。 因 为 
boolean 值 只 能 为 true 或 false,“ 大 于 ”和 “小 于 ”没有 实际 意义 。 
3.7.1 测试 对 象 的 等 价 性 
关系 操作 符 == 和 != 也 适用 于 所 有 对 象 ， 但 这 两 个 操作 符 通常 会 使 第 一 次 接触 Java 的 程序 员 
感到 迷惑 。 下 面 是 一 个 例子 : 
//: operators/Equivalence, java 
public class Equivalence { 
public static void main(String{] args) { 
Integer nl = new Integer (47); 
Integer n2 = new Integer (47); 


System.out.printIn(nl == n2); 
System.out.printin(nl != n2); 


} 
} /* Output: 
false 
true 
Wha 


语句 System.outprintin(nl == n2) 将 打印 出 括号 内 的 比较 式 的 布尔 值 结果 。 读 者 可 能 认为 输 
出 结果 肯定 先是 tue， 再 是 false， 因 为 两 个 Integer 对 象 都 是 相同 的 。 但 是 尽管 对 象 的 内 容 相同 ， 
然而 对 象 的 引用 却 是 不 同 的 ， 而 == 和 != 比 较 的 就 是 对 象 的 引用 。 所 以 输出 结果 实际 上 先是 false， 
再 是 tue。 这 自然 会 使 第 一 次 接触 关系 操作 符 的 人 感到 惊奇 。 

如 果 想 比较 两 个 对 象 的 实际 内 容 是 否 相 同 ， 又 该 如 何 操作 呢 ? 此 时 ， 必 须 使 用 所 有 对 象 都 
适用 的 特殊 方法 equalsO0。 但 这 个 方法 不 适用 于 “基本 类 型 "， 基 本 类 型 直接 使 用 == 和 != 即 可 。 
下 面 举例 说 明 如 何 使 用 : 


//: operators/EqualsMethod. java 


public class EqualsMethod { 
public static void main(String{] args) { 
Integer nl = new Integer (47): 
Integer n2 = new Integer (47): 
System.out.printin(n1.equals(n2)); 
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} 
} /* Output: 
true 
Mix 


结果 正如 我 们 所 预料 的 那样 。 但 事情 并 不 总 是 这 么 简单 1 假设 你 创建 了 自己 的 类 ， 就 像 下 
Wik. 
/1/: operators/EqualsMethod2. java 


// Default equals() does not compare contents. 


class Value { 
int i; 
$. 


public class EqualsMethod2 { 
public static void main(String[] args) { 
Value vi = new Value(); 
Value v2 = new Vatue(): 
v1.1 = v2.1 = 100; 
System. out.printIn(v1.equals(v2)) ; 


$ 
} /* Output: 
false 
“Ii~ 


事情 再 次 变 得 令 人 费解 了 : 结果 又 是 falsel 这 是 由 于 equals0 的 默认 行为 是 比较 引用 。 所 以 
除非 在 自己 的 新 类 中 覆盖 equals0 方 法 ， 否 则 不 可 能 表现 出 我 们 希望 的 行为 。 

遗民 的 是 ， 我 们 要 到 第 7 章 才学 习 覆 盖 ， 到 第 17 章 才学 习 如 何 恰当 地 定义 equals0。 但 在 这 之 
前 ， 请 留意 equals0 的 这 种 行为 表现 方式 ， 这 样 或 许 能 够 避免 一 些 “ 灾 难 ”。 

大 多 数 Java 类 库 都 实现 了 equals0 方 法 ， 以 便 用 来 比较 对 象 的 内 容 ， 而 非 比较 对 象 的 引用 。 

练习 5: (2) 创建 一 个 名 为 Dog 的 类 ， 它 包含 两 个 String 域 ， name 和 says。 在 main() 方 法 中 ， 
创建 两 个 Dog 对 象 ， 一 个 名 为 spot ( 它 的 叫 声 为 “Ruff! ")， 另 一 个 名 为 scruffy ( 它 的 叫 声 为 
“Wurf! “)。 然 后 显示 它们 的 名 字 和 叫 声 。 

练习 6，(3) 在 练习 5 的 基础 上 ， 创 建 一 个 新 的 Dog 索 引 ， 并 对 其 赋值 为 spot 对 象 。 测 试用 == 
和 equals0 方 法 来 比较 所 有 引用 的 结果 。 


3.8 逻辑 操作 符 


逻辑 操作 符 “ 与 ”(&&)、“ 或 ”( 目 、“ 非 ”(!0) 能 根据 参数 的 逻辑 关系 ， 生 成 一 个 布尔 值 
(troe 或 false) 。 下 面 这 个 例子 就 使 用 了 关系 操作 符 和 逻辑 操作 符 。 


//: operators/Bool. java 

// Relational and logical operators. 
import java.util.*; 

import static net.mindview.util.Print.*; 


public class Bool { 
public static void main(String[) args) { 
Random rand = new Random(47); 
int 1 = rand.nextInt (106) ; 
int j = rand.nextInt (168) ; 
print("i =" + i); 
print("j =" + 4): 
print("i > j is “+ (i > jD): 
print("1 < j is * 





print(" jis" 
print( is i 
print(* is" + (i == j); 


print" t= j is" + Gi t= j); 
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// Treating an int as a boolean is not legal Java: 
LE print( && j is * + (i && j)); 
/AL printi |] j is "+ G LJD 
Wt print(thi is "+ 11); 
print("(i < 10) && (j < 18) is " 
+ (Ci < 10) && (j < 18)) ); 
print("(i < 10) || (j < 18) is * 
+ (G < 10) I] (j < 19)) ); 


is true 

is false 
is true 
is false 





is false 

l= j is true 

(1 < 10) && (j < 10) is false 
(i < 10) |] (j < 10) is false 
LA 


“与 "、“ 或 "、“ 非 ”操作 只 可 应 用 于 布尔 值 。 与 在 C 及 C++ 中 不 同 的 是 : 不 可 将 一 个 非 布尔 
值 当 作 布尔 值 在 逻辑 表达 式 中 使 用 。 在 前 面 的 代码 中 用 “//!” 注 释 掉 的 语句 ， 就 是 错误 的 用 法 
(这 种 注释 语法 使 得 注释 能 够 被 自动 移 除 以 方便 测试 )。 后 面 的 两 个 表达 式 先 使 用 关系 比较 运算 ， 
生成 布尔 值 ， 然 后 再 对 产生 的 布尔 值 进行 逻辑 运算 。 

注意 ， 如 果 在 应 该 使 用 String 值 的 地 方 使 用 了 布尔 值 ， 布 尔 值 会 自动 转换 成 适当 的 文本 
形式 。 

在 上 述 程序 中 ， 可 将 整数 类 型 替换 成 除 布尔 型 以 外 的 其 他 任何 基本 数据 类 型 。 但 要 注意 ， 
对 浮 点 数 的 比较 是 非常 严格 的 。 即 使 一 个 数 仅 在 小 数 部 分 与 另 一 个 数 存在 极 微小 的 差异 ， 仍 然 
认为 它们 是 “不 相等 ”的 。 即 使 一 个 数 只 比 零 大 一 点 点 ， 它 仍然 是 “ 非 零 ” 值 。 

练习 7: (3) 编 写 一 个 程序 ， 模 拟 扔 硬币 的 结果 。 


3.8.1 短路 

当 使 用 逻辑 操作 符 时 ， 我 们 会 遇 到 一 种 “短路 ”现象 。 即 一 旦 能 够 明确 无 误 地 确定 整个 表 
达 式 的 值 ， 就 不 再 计算 表达 式 余下 部 分 了 。 因 此 ， 整 个 逻辑 表达 式 靠 后 的 部 分 有 可 能 不 会 被 运 
算 。 下 面 是 演示 短路 现象 的 例子 : 


41: operators/ShortCircuit.java 
// Demonstrates short-circuiting behavior 
// with logical operators. 

import static net.mindview.util.Print.*; 


public class ShortCircuit { 
static boolean test1(int val) { 
print("testi(" + val + ")"); 
print("result: " + (val < 1)); 
return val < 1; 


} 

static boolean test2(int val) { 
print("test2(" + val + ")"); 
print("result: " + (val < 2)); 
return val < 2; 

} 

static boolean test3(int val) { 
print("test3(" + val + ")"); 
print("result: " + (val < 3)); 
return val < 3; 

} 
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public static void main(String[] args) { 
boolean b = test1(0) && test2(2) && test3(2); 
print ("expression is ”+ b); 


} 
} /* Output: 
test1(0) 
result: true 
test2(2) 
result: false 
expression is false 
Ii~ 


每 个 测试 都 会 比较 参数 ， 并 返回 true 或 false。 它 也 会 打印 信息 告诉 你 正在 调用 测试 。 这 些 测 
试 都 作用 于 下 面 这 个 表达 式 : 

test1(0) && test2(2) && test3(2) 

你 会 很 自然 地 认为 所 有 这 三 个 测试 都 会 得 以 执行 。 但 输出 显示 却 并 非 这 样 。 第 一 个 测试 生成 
结果 true， 所 以 表达 式 计算 会 继续 下 去 。 然 而 ， 第 二 个 测试 产生 了 一 个 false 结 果 。 由 于 这 意味 着 
整个 表达 式 肯 定 为 false， 所 以 没 必要 继续 计算 剩余 的 表达 式 ， 那 样 只 是 浪费 。“ 短 路 ”一 词 的 由 
来 正 源 于 此 。 事 实 上 ， 如 果 所 有 的 逻辑 表达 式 都 有 一 部 分 不 必 计算 ， 那 将 获得 潜在 的 性 能 提升 。 


3.9 直接 常量 


一 般 说 来 ， 如 果 在 程序 里 使 用 了 “直接 常量 "， 编 译 器 可 以 准确 地 知道 要 生成 什么 样 的 类 型 ， 
但 有 时 候 却 是 模棱两可 的 。 如 果 发 生 这 种 情况 ， 必 须 对 编译 器 加 以 适当 的 “指导 "， 用 与 直接 量 
相关 的 某 些 字符 来 额外 增加 一 些 信息 。 下 面 这 段 代码 向 大 家 展示 了 这 些 字符 。 


//: operators/Literals. java 
import static net.mindview.util.Print.*; 


public class Literals { 

public static void main(String(] args) { 
int 11 = @x2f; // Hexadecimal (lowercase) 
print("{1: " + Integer. toBinaryString(i1)); 
int i2 = 6X2F; // Hexadecimal (uppercase) 
print("42: " + Integer. toBinaryString(i2)); 
int i3 = 0177; // Octal (leading zero) 
print("i3: " + Integer. toBinaryString(13)); 
char c = Oxffff; // max char hex value 
print("c: " + Integer. toBinaryString(c)); 





byte b = @x7f; // max byte hex value 
print("b: " + Integer. toBinaryString(b)); 

short s = Ox7fff; // max short hex value 

print("s: ”+ Integer. toBinaryString(s)) ; 

long nl = 296L; // long suffix 

long n2 = 2601; // long suffix (but can be confusing) 
long n3 = 260; 

float f1 = 1; 

float f2 = 1F; // float suffix 

float f3 = 1f; // float suffix 

double d1 = 1d; // double suffix 

double d2 = 1D; // double suffix 

// (Hex and Octal also work with long) 





} 
} /* Output: 
i1: 101111 
i2: 101111 
43: 1111111 
c: 1111111111111111 
b:, 1111111 
$: 111111111111111 
I~ 
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直接 常量 后 面 的 后 级 字符 标志 了 它 的 类 型 。 若 为 大 写 (或 小 写 ) 的 L， 代 表 long (但 是 ， 使 
用 小 写字 母 ! 容 易 造成 混淆 ， 因 为 它 看 起 来 很 像 数 字 1)。 大 写 (或 小 写 ) 字母 F， 代 表 float 大 
写 (或 小 写 ) 字母 D， 则 代表 double。 

十 六 进 制 数 适用 于 所 有 整数 数据 类 型 ， 以 前 绥 0x (或 0X)， 后 面 跟随 0-9 或 小 写 (或 大 写 ) 
的 a-f 来 表示 。 如 果 试 图 将 一 个 变量 初始 化 成 超出 自身 表示 范围 的 值 (无 论 这 个 值 的 数值 形式 如 
何 ) ， 编 译 器 都 会 向 我 们 报告 一 条 错误 信息 。 注 意 在 前 面 的 代码 中 ， 已 经 给 出 了 char、byte 以 及 
short 所 能 表示 的 最 大 的 十 六 进 制 值 。 如 果 超出 范围 ， 编 译 器 会 将 值 自动 转换 成 int 型 ， 并 告诉 我 
们 需要 对 这 次 赋值 进行 “ 窗 化 转型 ”( 转 型 将 在 本 章 稍 后 部 分 定义 )。 这 样 我 们 就 可 清楚 地 知道 
自己 的 操作 是 否 越界 了 。 

八进制 数 由 前 缀 0 以 及 后 续 的 0~7 的 数字 来 表示 。 

在 C、C++ 或 者 Java 中 ， 二 进 制 数 没有 直接 常量 表示 方法 。 但 是 ， 在 使 用 十 六 进 制 和 八进制 
记 数 法 时 ， 以 二 进 制 形式 显示 结果 将 非常 有 用 。 通 过 使 用 Integer 和 Long 类 的 静态 方法 
toBinaryString() 可 以 很 容易 地 实现 这 一 点 。 请 注意 ， 如 果 将 比较 小 的 类 型 传递 给 Integer. 
toBinaryString0 方 法 ， 则 该 类 型 将 自动 被 转换 为 int。 

练习 8: (2) 展示 用 十 六 进 制 和 八进制 记 数 法 来 操作 long 值 ， 用 Long.toBinaryString0 来 显示 
结果 。 

3.9.1 指数 记 数 法 

Java 采 用 了 一 种 很 不 直观 的 记 数 法 来 表示 指数 ， 例 如 

//: operators/Exponents. java 

// "e" means “10 to the power.” 


public class Exponents { 
public static void main(String[] args) { 
// Uppercase and lowercase 'e' are the same: 
float expFloat = 1.39e-43f; 
expFloat = 1.39€-43f; 
System. out. println(expFloat) ; 
double expDouble = 47e47d; // 'd' is optional 
double expDouble2 = 47e47; // Automatically double 
System. out.print1n(expDouble) ; 
} 
} /* Output: 
1,39E-43 
4.7648 
Wha 


在 科学 与 工程 领域 ,，“e” 代 表 自 然 对 数 的 基数 ， 约 等 于 2.718 (Java 中 的 Math.E 给 出 了 更 精 
确 的 double 型 的 值 )。 例 如 1.39 xe“ 这 样 的 指数 表达 式 意味 着 1.39 x 2.718%, RT, BELT 
FORTRAN 语 言 的 时 候 ， 设 计 师 们 很 自然 地 决定 e 代 表 “10 的 等 次 "。 这 种 做 法 很 奇怪 ， 因 为 
FORTRAN 最 初 是 面向 科学 与 工程 设计 领域 的 ， 它 的 设计 者 们 对 引入 这 样 容易 令 人 混淆 的 概念 应 


该 很 敏感 才 对 。 。 但 不 管 怎样 ， 这 种 惯例 在 C、C++ 以 及 Java 中 被 保留 了 下 来 。 所 以 倘若 习惯 将 e 


© John Kirkham Silt: “我 开始 用 计算 机 是 在 1962 年 ， 使 用 的 是 IBM 1620 机 器 上 的 FORTRAN II。 那 时 候 ， 从 60 
年 代 到 70 年 代 ，FORTRAN 一 直 都 是 使 用 大 写字 母 。 之 所 以 会 出 现 这 一 情况 ， 可 能 是 由 于 早期 的 输入 设备 大 多 
是 老式 电 传 打字 机 ， 使 用 5 位 Baudot 码 ， 那 是 不 包括 小 写字 母 的 。 乘 冠 表达 式 中 的 “E” 也 肯定 是 大 写 的 ， 所 
以 不 会 与 自然 对 数 的 基数 “e” 发 生 冲 突 ， 后 者 必然 是 小 写 的 。“E” 这 个 字母 的 含义 其 实 很 简单 ， 就 是 
“Exponential” 的 意思 ， 即 “指数 ”或 “等 数 "， 代 表 数 字 系统 的 基数 一 -一般 都 是 10。 当 时 ， 八 进 制 也 在 程序 
员 中 广泛 使 用 。 尽 管 我 自己 未 看 到 它 的 使 用 ， 但 假若 我 在 乘 里 表达 式 中 看 到 一 个 八进制 数字 ， 我 就 会 认为 基 
数 是 8。 我 记得 第 一 次 看 到 用 小 写 “e” 表 示 指 数 是 在 70 年 代 末 期 。 我 当时 也 觉得 它 极 易 产生 混淆 。 产 生 这 个 
问题 是 因为 FORTRAN 已 经 渐渐 引入 了 小 写字 母 ， 但 并 非 一 开始 就 有 。 如 果 你 真 的 想 使 用 自然 对 数 的 基数 ， 有 
现成 的 函数 可 供 利 用 ， 但 它们 都 是 大 写 的 .” 
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作为 自然 对 数 的 基数 使 用 ， 那 么 在 Java 中 看 到 像 1.39e“f 这 样 的 表达 式 时 ， 请 转换 思维 ， 它 真正 
的 含义 是 1.39 x 10, 
注意 如 果 编 译 器 能 够 正确 地 识别 类 型 ， 就 不 必 在 数值 后 附加 字符 。 例 如 语句 
long n3 = 260; 
它 不 存在 含混 不 清 的 地 方 ， 所 以 200 后 面 的 L 是 用 不 着 的 。 然 而 ， 对 于 语句 110 
float f4 = le-43f; // 10 to the power 
编译 器 通常 会 将 指数 作为 双 精 度数 《double) 处 理 ， 所 以 假如 没有 这 个 尾随 的 f， 就 会 收 到 一 条 
出 错 提示 ， 告 诉 我 们 必须 使 用 类 型 转换 将 double 转 换 成 float。 
练习 9: (1) 分 别 显 示 用 float 和 double 指 数 记 数 法 所 能 表示 的 最 大 和 最 小 的 数字 。 


3.10 按 位 操作 符 


按 位 操作 符 用 来 操作 整数 基本 数据 类 型 中 的 单个 “比特 ”(bit) ， 即 二 进 制 位 。 按 位 操作 符 
会 对 两 个 参数 中 对 应 的 位 执行 布尔 代数 运算 ， 并 最 终生 成 一 个 结果 。 

按 位 操作 符 来 源 于 C 语 言 面向 底层 的 操作 ， 在 这 种 操作 中 经 常 需要 直接 操纵 硬件 ， 设 置 硬件 
寄存 器 内 的 二 进 制 位 。Java 的 设计 初衷 是 做 入 电视 机 机 顶 盒 内 ， 所 以 这 种 面向 底层 的 操作 仍 被 
保留 了 下 来 。 但 是 ， 人 们 可 能 不 会 过 多 地 用 到 位 操作 符 。 

如 果 两 个 输入 位 都 是 1， 则 按 位 “与 ”操作 符 (Se) 生成 一 个 输出 位 1， 否 则 生成 一 个 输出 
位 0。 如 果 两 个 输入 位 里 只 要 有 一 个 是 1， 则 按 位 “或 ”操作 符 (1) 生成 一 个 输出 位 1， 只 有 在 两 
个 输入 位 都 是 0 的 情况 下 ， 它 才 会 生成 一 个 输出 位 0。 如 果 输 入 位 的 某 一 个 是 1， 但 不 全 都 是 1， 
那么 按 位 “ 异 或 ”操作 (^) 生成 一 个 输出 位 1。 按 位 “ 非 ”(~)， 也 称 为 取 反 操作 符 ， 它 属于 一 
元 操作 符 ， 只 对 一 个 操作 数 进行 操作 (其 他 按 位 操作 符 是 二 元 操作 符 )。 按 位 “ 非 ” 生 成 与 输入 
位 相反 的 值 一 车 输 入 0， 则 输出 1， 若 输入 1， 则 输出 0。 

按 位 操作 符 和 逻辑 操作 符 都 使 用 了 同样 的 符号 ， 因 此 我 们 能 方便 地 记 住 它 们 的 含义 ;由 于 
位 是 非常 “小 ”的 ， 所 以 按 位 操作 符 仅 使 用 了 一 个 字符 。 

按 位 操作 符 可 与 等 号 (=) 联合 使 用 ， 以 便 合 并 运算 和 赋值 ， &=、|= 和 ^= 都 是 合法 的 〈 由 于 
“~” 是 一 元 操作 符 ， 所 以 不 可 与 “=” 联 合 使 用 ) 。 

我 们 将 布尔 类 型 作为 一 种 单 比特 值 对 待 ， 所 以 它 多 少 有 些 独特 。 我 们 可 对 它 执行 按 位 “与 "、 [1] 
按 位 “或 ”和 按 位 “ 异 或 ”运算 ， 但 不 能 执行 按 位 “ 非 ”( 大 概 是 为 了 避免 与 逻辑 NOT 混 消 ) 。 
对 于 布尔 值 ， 按 位 操作 符 具有 与 逻辑 操作 符 相 同 的 效果 ， 只 是 它们 不 会 中 途 “ 短 路 "。 此 外 ， 针 
对 布尔 值 进行 的 按 位 运算 为 我 们 新 增 了 一 个 “ 异 或 ”逻辑 操作 符 ， 它 并 未 包括 在 “逻辑 ”操作 
符 的 列表 中 。 在 移 位 表达 式 中 ， 不 能 使 用 布尔 运算 ， 原 因 将 在 后 面 解释 。 

练习 10: (3) 编写 一 个 具有 两 个 常量 值 的 程序 ， 一 个 具有 交替 的 二 进 制 位 1 和 0， 其 中 最 低 有 
效 位 为 0， 另 一 个 也 具有 交替 的 二 进 制 位 1 和 0， 但 是 其 最 低 有 效 位 为 1 (提示 : 使 用 十 六 进 制 常 
量 来 表示 是 最 简单 的 方法 )。 取 这 两 个 值 ， 用 按 位 操作 符 以 所 有 可 能 的 方式 结合 运算 它们 ,然后 
用 IntegertoBinaryString0 显 示 。 


3.11 移 位 操作 符 


移 位 操作 符 操作 的 运算 对 象 也 是 二 进 制 的 “位 "。 移 位 操作 符 只 可 用 来 处 理 整 数 类 型 (基本 
类 型 的 一 种 )。 左 移 位 操作 符 (<<) 能 按照 操作 符 右 侧 指定 的 位 数 将 操作 符 左 边 的 操作 数 向 左 移 
动 (在 低位 补 0)。“ 有 符号 ” 右 移 位 操作 符 (>>) 则 按照 操作 符 右 侧 指定 的 位 数 将 操作 符 左边 的 
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PPM). AFS HERETER “SPR: 车 符 号 为 正 ， 则 在 高 位 插入 0， 若 
符号 为 负 ， 则 在 高 位 插入 1。Java 中 增加 了 一 种 “无 符号 ” 右 移 位 操作 符 (>>>)， 它 使 用 “ 零 扩 
FR”: 无 论 正 负 ， 都 在 高 位 插入 0。 这 一 操作 符 是 C 或 C++ 中 所 没有 的 。 

如 果 对 char、byte 或 者 short 类 型 的 数值 进行 移 位 处 理 ， 那 么 在 移 位 进行 之 前 ， 它 们 会 被 转 
换 为 int 类 型 ， 并 且 得 到 的 结果 也 是 一 个 int 类 型 的 值 。 只 有 数值 右 端的 低 5 位 才 有 用 。 这 样 可 防 
止 我 们 移 位 超过 int 型 值 所 具有 的 位 数 。( 译 注 :因为 2 的 5 次 方 为 32， 而 int 型 值 只 有 32 位 。) 车 对 
一 个 long 类 型 的 数值 进行 处 理 ， 最 后 得 到 的 结果 也 是 long。 此 时 只 会 用 到 数值 右 端的 低 6 位 ， 以 
防止 移 位 超过 long 型 数值 具有 的 位 数 。 

“ 移 位 ”可 与 “等 号 ”(<<= 或 >>= 或 >>>=) 组 合 使 用 。 此 时 ， 操 作 符 左边 的 值 会 移动 由 右边 
的 值 指定 的 位 数 ， 再 将 得 到 的 结果 赋 给 左边 的 变量 。 但 在 进行 “无 符号 ” 右 移 位 结合 赋值 操作 
时 ， 可 能 会 遇 到 一 个 问题 : 如 果 对 byte 或 short 值 进行 这 样 的 移 位 运算 ， 得 到 的 可 能 不 是 正确 的 
结果 。 它 们 会 先 被 转换 成 int 类 型 ， 再 进行 右 移 操作 ， 然 后 被 截断 ， 赋 值 给 原来 的 类 型 ， 在 这 种 
情况 下 可 能 得 到 -1 的 结果 。 下 面 这 个 例子 演示 了 这 种 情况 : 


//: operators/URShift .java 
// Test of unsigned right shift. 
import static net.mindview.util.Print.*; 


public class URShift { 
public static void main(String[] args) { 

int 1 = -1; 
print (Integer. toBinaryString(i)); 
1 >>= 10; 
print(Integer. toBinaryString(i)); 
long 1 = -1; 
print (Long. toBinaryString(1)): 
1 >>>= 10; 
print (Long. toBinaryString(1)); 
short s = -1; 
print(Integer. toBinaryString(s)); 
s >>= 16; 
print (Integer. toBinaryString(s)); 
byte b = -1; 
print (Integer. toBinaryString(b)); 
b >>>= 10; 
print (Integer. toBinaryString(b)); 
b = -1; 
print (Integer. toBinaryString(b)); 
print (Integer. toBinaryString(b>>>19)) ; 


} 
} /* Output: 
11111111111111111111111111111111 
1111111111111111111111 
11411111111111111111111111111111111111111111111111111111111111111 
11111111111111111111111111111111111111111111111111111 
11111111211111111111111111111111 
11111111111111111111111111111111 
11111111111111111111111111111111 
11111111111111111111111111111111 
11111111111111111111111111111111 
1113111111111111111111 
:~ 


在 最 后 一 个 移 位 运算 中 ， 结 果 没 有 赋 给 b， 而 是 直接 打印 出 来 ， 所 以 其 结果 是 正确 的 。 
下 面 这 个 例子 向 大 家 阑 示 了 如 何 应 用 涉及 “ 按 位 ”操作 的 所 有 操作 符 ， 


//: operators/BitManipulation. java 

// Using the bitwise operators. 

import java.util.*; 

import static net.mindview.util.Print.*; 





=] 
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public class BitManipulation { 

public static void main(String[] args) { 
Random rand = new Random(47) ; 
int i = rand.nextInt(): 
int j = rand.nextInt() 
printBinaryInt("-1", -1); 
printBinaryInt("+1", +1) 
int maxpos = 2147483647; 
printBinaryInt("maxpos", maxpos); 
int maxneg = -2147483648; 
printBinaryInt("maxneg", maxneg); 
printBinaryInt("i", i); 
printBinaryInt("~i", ~i 
printBinaryInt("-i", -i 















printBinaryInt("j", j) 
printBinaryInt("i & j 1&j): 
printBinaryInt("i | j". i | j): 
printBinaryInt("i ^ j", i ^j); 


printBinaryInt("i << 5", i << 5); 
printBinaryInt("i >> 5", 1 >> 5); 
printBinaryInt("(~i) >> 5", (~i) >> 5); 
printBinaryInt("i >>> 5", 1 >>> 5); 
printBinaryInt("(~i) >>> 5", (~i) >>> 5); 


long 1 = rand.nextLong(); 
long m = rand.nextLong( 
printBinaryLong("-1L", -1L); 
printBinarytong("+iL", +1L); 
long 11 = 9223372036854775807L; 
printBinaryLong("maxpos”. 11); 
long lin = -9223372636854775808L; 
printBinaryLong(*maxneg”, lin): 
printBinaryLong("t", 1) 
printBinaryLong(*~1", ~1) 
printBinaryLong("-1", -1): 
printBinaryLong("m", m); 

















printBinaryLong("l & m", l & m); 
printBinaryLong("t | m", 1 | m) 
printBinaryLong("t ^ m", 1 ^m 





printBinaryLong("l << 5 
printBinaryLong("l >> 5", 1 >> 5); 
printBinarylong("(~1) >> 5", (~U >> 5); 
printBinaryLong("1 >>> 5", 1 >>> 5); 
printBinaryLong("(~1) >>> 5", (~1) >>> 5); 

} 

Static void printBinaryInt(String s, int i) { 
print(s +", int: "+ i+", binary:\n ”+ 

Integer. toBinaryString(i)); 





} 
static void printBinaryLong(String s, long 1) { 
print(s +", long: "+ 1+", binary:\n "+ 
Long. toBinaryString(1)): 

} 

} /* Output: 

-1, int: -1, binary: 
11111111111111111111111111111111 

+1, int: 1, binary: 
1 

maxpos, int: 2147483647, binary: 
1111111111111111111111111111111 

maxneg, int: -2147483648, binary: 
10000000000000000000000000000090 

1, int: -1172028779, binary: 
10111010001001000100001010010101 

~i, int: 1172028778, binary: 
1666161119116111611116161161919 

<i, int: 1172028779, binary: 
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109010111011111011110161101611 
j. int: 1717241110, binary: 
1100110010110110000010100010110 
1 & j, int: 570425364, binary: 
100010000000000000600000916100 
1 lj. int: -25213033, binary: 
11111110011111110100011116010111 
i ^j, int: -595638397, binary: 
11011100011111110100011110000011 
i << 5, int: 1149784736, binary: 
1000100100010000101001010106009 
i >> 5, int: -36625996, binary: 
1111110111610¢619010001000010100 
(~1) >> 5, int: 36625899, binary: 
19901011101101110111101011 
i >>> 5, int: 97591828, binary: 
101110100010010001000010100 
(~i) >>> 5, int: 36625899, binary: 
10001011101101110111101911 


Whi~ 

程序 末尾 调用 了 两 个 方法 ， printBinaryInt0 和 printBinaryLong0。 它 们 分 别 接受 int 或 long 
型 的 参数 ， 并 用 二 进 制 格式 输出 ， 同 时 附 有 简要 的 说 明文 字 。 上 面 的 例子 还 展示 了 对 int 和 long 
的 所 有 按 位 操作 符 的 作用 ， 还 展示 了 int 和 long 的 最 小 值 、 最 大 值 、+1 和 -1 值 ， 以 及 它们 的 二 进 
制 形式 ， 以 使 大 家 了 解 它 们 在 机 器 中 的 具体 形式 。 注 意 最 高 位 表示 符号 ;0 为 正 ，1 为 负 。 关 于 
int 部 分 的 输出 正如 上 面 所 示 。 

数字 的 二 进 制 表示 形式 称 为 “有 符号 的 二 进 制 补 码 "。 

练习 11: (3) 以 一 个 最 高 有 效 位 为 1 的 二 进 制 数字 开始 (提示 : 使 用 十 六 进 制 常 量 )， 用 有 符 
号 右 移 操作 符 对 其 进行 右 移 ， 直 至 所 有 的 二 进 制 位 都 被 移出 为 止 ， 每 移 一 位 都 要 使 用 
JntegertoBinaryString0 显 示 结 果 。 

练习 12: (3) 以 一 个 所 有 位 都 为 1 的 二 进 制 数字 开始 ， 先 左 移 它 ， 然 后 用 无 符号 右 移 操作 符 
对 其 进行 右 移 ， 直 至 所 有 的 二 进 制 位 都 被 移出 为 止 ， 每 移 一 位 都 要 使 用 Integer.toBinaryString0 
显示 结果 。 

练习 13:(1) 编写 一 个 方法 ， 它 以 二 进 制 形式 显示 char 类 型 的 值 。 使 用 多 个 不 同 的 字符 来 展 
示 它 。 


3.12 三 元 操作 符 if-else 


三 元 操作 符 也 称 为 条 件 操作 符 ， 它 显得 比较 特别 ， 因 为 它 有 三 个 操作 数 ， 但 它 确实 属于 操 
作 符 的 一 种 ， 因 为 它 最 终 也 会 生成 一 个 值 ， 这 与 本 章 下 一 节 中 介绍 的 普通 这 else 语 句 是 不 同 的 。 
其 表达 式 采 取 下 述 形式 : 

boolean-exp ? value@ : valuel 

如 果 boolean-exp (布尔 表达 式 ) 的 结果 为 true， 就 计算 value0， 而 且 这 个 计算 结果 也 就 是 操 
作 符 最 终 产 生 的 值 。 如 果 boolean-exp 的 结果 为 false， 就 计算 value1， 同 样 ， 它 的 结果 也 就 成 为 了 
操作 符 最 终 产 生 的 值 。 

当然 ， 也 可 以 换 用 普通 的 if-else 语 句 (在 后 面 介 绍 ) ， 但 三 元 操作 符 更 加 简洁 。 尽 管 C (Ch 
发 明了 该 操作 符 ) 引 以 为 做 的 就 是 它 是 一 种 简练 的 语言 ， 而 且 三 元 操作 符 的 引入 多 半 就 是 为 了 
体现 这 种 高 效率 的 编程 ， 但 假如 你 打算 频繁 使 用 它 ， 还 是 要 多 作 思 量 ， 因 为 它 很 容易 产生 可 读 
性 极 差 的 代码 。 

条 件 操作 符 与 if-else 完 全 不 同 ， 因 为 它 会 产生 一 个 值 。 下 面 是 这 两 者 进行 比较 的 示例 : 
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//: operators/TernaryIfElse. java 
‘import static net.mindview.util.Print.*; 


public class TernaryIfElse { 
static int ternary(int i) { 
return i < 10? i * 100 : i * 10; 


dtatte int standardIfElse(int i) { 
if(i < 10) 
return i * 160; 
else 
return i * 10; 


public static void main(String{] args) { 
print(ternary(9)); 
print(ternary(16)); 
print (standardIfElse(9)); 
print (standardIfElse(10)); 
} 
} /* Output: 
960 


100 
900 
109 
Whim 


可 以 看 出 ， 上 面 的 ternary0 中 的 代码 与 standardIfElse0 中 不 用 三 元 操作 符 的 代码 相 比 ， 显 
得 更 加 紧凑 ， 但 standardIfElse0 更 易 理解 ， 而 且 不 需要 太 多 的 录入 。 所 以 在 选择 使 用 三 元 操作 
符 时 ， 请 务必 仔细 考虑 。 


3.13 字符 囊 操作 符 + 和 += 


这 个 操作 符 在 Java 中 有 一 项 特殊 用 途 : 连接 不 同 的 字符 串 。 这 一 点 已 经 在 前 面 的 例子 中 展 
示 过 了 。 尽 管 与 + 和 += 的 传统 使 用 方式 不 太一 样 ， 但 我 们 还 是 很 自然 地 使 用 这 些 操作 符 来 做 这 
件 事情 。 

这 项 功能 用 在 C++ 中 似乎 是 个 不 错 的 主意 ， 所 以 引入 了 操作 符 重 载 (operator overloading) 
机 制 ， 以 便 C++ 程序 员 可 以 为 几乎 所 有 操作 符 增加 功能 。 但 非常 遗憾 ， 与 C++ 的 另外 一 些 限制 结 
合 在 一 起 ， 使 得 操作 符 重 载 成 为 了 一 种 非常 复杂 的 特性 ， 程 序 员 在 设计 自己 的 类 时 必须 对 此 有 
非常 周全 的 考虑 。 与 C++ 相 比 ， 尽 管 操作 符 重 载 在 Java 中 更 易 实现 (就 像 在 Ch 语言 中 所 展示 的 那 
样 ， 它 具有 相当 简单 直接 的 操作 符 重 载 机 制 )， 但 仍然 过 于 复杂 。 所 以 Java 程 序 员 不 能 像 C++ 和 
C# 程 序 员 那 样 实现 自己 的 重 载 操作 符 。 

字符 串 操作 符 有 一 些 很 有 趣 的 行为 。 如 果 表 达 式 以 一 个 字符 串 起 头 ， 那 么 后 续 所 有 操作 数 
都 必须 是 字符 串 型 〈 请 记 住 ， 编 译 器 会 把 双 引 号 内 的 字符 序列 自动 转 成 字符 串 ) : 

//: operators/StringOperators.java 

import static net.mindview.util.Print.*: 


public class StringOperators { 
public static void main Strinati args) { 

int x= 8, y= 4, z= 2; 
String s = "x, y, z"; 
print(s + x +y +z); 
print(x + " * + s); // Converts x to a String 
s += "(summed) = "; // Concatenation operator 
print(s + (x + y + z)); 
print("" + x); // Shorthand for Integer. toString() 


} 
} /* Output: 
x. y, z 612 
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Ox, y, z 
x. y. Z (summed) = 3 


o 
When 


请 注意 ， 第 一 个 打印 语句 的 输出 是 012 而 不 是 3， 而 3 正 是 将 这 些 整 数 求 和 之 后 应 该 得 到 的 结 
果 ， 之 所 以 出 现 这 种 情况 ， 是 因为 Java 编 译 器 会 将 x、y 和 z 转 换 成 它们 的 字符 串 形式 ， 然 后 连接 
这 些 字符 种， 而 不 是 先 把 它们 加 到 一 起 。 第 二 个 打印 语句 把 先导 的 变量 转换 为 String， 因 此 这 个 
字符 串 转换 将 不 依赖 于 第 一 个 变量 的 类 型 。 最 后 ， 可 以 看 到 使 用 += 操 作 符 将 一 个 字符 串 追加 到 
了 s 上 ， 并 且 使 用 了 括号 来 控制 表达 式 的 赋值 顺序 ， 以 使 得 int 类 型 的 变量 在 显示 之 前 确实 进行 了 
求 和 操作 。 

请 注意 main0 中 的 最 后 一 个 示例 ， 有 时 会 看 到 这 种 一 个 空 的 String 后 面 跟随 + 和 一 个 基本 类 
型 变量 ， 以 此 作为 不 调用 更 加 麻烦 的 显 式 方法 〈 在 本 例 中 应 该 是 IntegertoString0) 而 执行 字符 
串 转 换 的 方式 。 


3.14 使 用 操作 符 时 常 犯 的 错误 


使 用 操作 符 时 一 个 常 犯 的 错误 就 是 ， 即 使 对 表达 式 如 何 计算 有 点 不 确定 ， 也 不 愿意 使 用 括 
号 。 这 个 问题 在 Java 中 仍然 存在 。 

在 C 和 C++ 中 ， 一 个 特别 常见 的 错误 如 下 : 

while(x = y) { 

ss 

} 

程序 员 很 明显 是 想 测试 是 否 “ 相 等 ”(==) ， 而 不 是 进行 赋值 操作 。 在 C 和 C++ 中 ， 如 果 y 是 
一 个 非 零 值 ， 那 么 这 种 赋值 的 结果 肯定 是 tue， 而 这 样 便 会 得 到 一 个 无 穷 循 环 。 在 Java 中 ， 这 个 
表达 式 的 结果 并 不 是 布尔 值 ， 而 编译 器 期 望 的 是 一 个 布尔 值 。 由 于 Java 不 会 自动 地 将 int 数 值 转 
换 成 布尔 值 ， 所 以 在 编译 时 会 抛 出 一 个 编译 时 错误 ， 从 而 阻止 我 们 进一步 去 运行 程序 。 所 以 这 
种 错误 在 Java 中 永远 不 会 出 现 (唯一 不 会 得 到 编译 时 错误 的 情况 是 x 和 y 都 为 布尔 值 。 在 这 种 情 

Tio] 况 下 ,x=y 属 于 合法 表达 式 。 而 在 前 面 的 例子 中 ， 则 可 能 是 一 个 错误 )。 

Java 中 有 一 个 与 C 和 C++ 中 类 似 的 问题 ， 即 使 用 按 位 “与 ”和 按 位 “或 ”代替 逻辑 “与 ”和 

逻辑 “或 "。 按 位 “与 ”和 按 位 “或 ”使 用 单字 符 (& 或 |) ， 而 逻辑 “与 ”和 逻辑 “或 ”使 用 双 
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字符 〈&& 或 0) 。 就 像 “=” 和 “==” 一 样 ， 键 人 一 个 字符 当然 要 比 键入 两 个 简单 。Java 编 译 器 
可 防止 这 个 错误 发 生 ， 因 为 它 不 允许 我 们 随便 把 一 种 类 型 当 作 另 一 种 类 型 来 用 。 
3.15 类 型 转换 操作 符 


类 型 转换 (cast) 的 原意 是 “模型 铸造 "。 在 适当 的 时 候 ，Java 会 将 一 种 数据 类 型 自动 转换 
成 另 一 种 。 例 如 ， 假 设 我 们 为 某 浮 点 变量 赋 以 一 个 整数 值 ， 编 译 器 会 将 int 自 动 转换 成 foat。 类 
型 转换 运算 允许 我 们 显 式 地 进行 这 种 类 型 的 转换 ， 或 者 在 不 能 自动 进行 转换 的 时 候 强制 进行 类 
型 转换 。 

要 想 执行 类 型 转换 ， 需 要 将 希望 得 到 的 数据 类 型 置 于 圆 括号 内 ， 放 在 要 进行 类 型 转换 的 值 
的 左边 ， 可 以 在 下 面 的 示例 中 看 到 它 : 

/1/: operators/Casting.java 


public class Casting { 
public static void main(String{] args) { 
int 1 = 200; 
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long 1ng = (long)i; 

Ing = i; // "Widening." so cast not really required 
long tng2 = (long)200; 

1ng2 = 200; 

1/ A “narrowing conversion": 

i = (int)lng2; // Cast required 


} 

WES 

正如 所 看 到 的 ， 既 可 对 数值 进行 类 型 转换 ， 亦 可 对 变量 进行 类 型 转换 。 请 注意 ， 这 里 可 能 
会 引入 “多 余 的 ”转型 ， 例 如 ， 编 译 器 在 必要 的 时 候 会 自动 进行 int 值 到 long 值 的 提升 。 但 是 你 
仍然 可 以 做 这 样 “ 多 余 的 ” 事 ， 以 提醒 自己 需要 留意 ， 也 可 以 使 代码 更 清楚 。 在 其 他 情况 下 ， 
可 能 只 有 先进 行 类 型 转换 ， 代 码 编译 才能 通过 。 

在 C 和 C++ 中 ， 类 型 转换 有 时 会 让 人 头痛 。 但 是 在 Java 中 ， 类 型 转换 则 是 一 种 比较 安全 的 操 
作 。 然 而 ， 如 果 要 执行 一 种 名 为 富 化 转换 (narrowing conversion) 的 操作 (也 就 是 说 ， 将 能 容纳 
更 多 信息 的 数据 类 型 转换 成 无 法 容纳 那么 多 信息 的 类 型 ), 就 有 可 能 面临 信息 丢失 的 危险 。 此 时 ， 
编译 器 会 强制 我 们 进行 类 型 转换 ， 这 实际 上 是 说 ;“ 这 可 能 是 一 件 危险 的 事情 ， 如 果 无 论 如 何 要 
这 么 做 ,必须 显 式 地 进行 类 型 转换 。” 而 对 于 扩展 转 接 (widening conversion)， 则 不 必 显 式 地 进 
行 类 型 转换 ， 因 为 新 类 型 肯定 能 容纳 原来 类 型 的 信息 ， 不 会 造成 任何 信息 的 丢失 。 

Java 克 许 我 们 把 任何 基本 数据 类 型 转换 成 别 的 基本 数据 类 型 ， 但 布尔 型 除外 ， 后 者 根本 不 多 
许 进行 任何 类 型 的 转换 处 理 。“ 类 ”数据 类 型 不 允许 进行 类 型 转换 。 为 了 将 一 种 类 转换 成 另 一 种 ， 
必须 采用 特殊 的 方法 〈 本 书后 面 会 讲 到 ， 对 象 可 以 在 其 所 属 类 型 的 类 族 之 间 可 以 进行 类 型 转换 ， 
例如 ,“ 橡 树 ”可 转型 为 “ 树 ”， 反之 亦 然 。 但 不 能 把 它 转换 成 类 族 以 外 的 类 型 ， 如 “岩石 ") 。 
3.15.1 RAGA 

在 执行 窗 化 转换 时 ， 必 须 注意 蕉 尾 与 伟人 问题 。 例 如 ， 如 果 将 一 个 浮 点 值 转换 为 整 型 值 ， 
Java 会 如 何 处 理 呢 ? 例如 ， 将 29.7 转 换 为 int， 结 果 是 30 还 是 29? 在 下 面 的 示例 中 可 以 找到 答案 : 


//: operators/CastingNumbers.java 

// What happens when you cast a float 
// or double to an integral value? 
import static net.mindview.util.Print.*; 


public class CastingNumbers { 

public static void main(String(] args) { 
double above = 0.7, below = 0.4; 
float fabove = 0.7f, fbelow = 0.4f; 
print("(int)above: ”+ (int)above); 
print("(int)below: * + (int)below); 
print("(int)fabove: " + (int)fabove); 
print("(int)fbelow: " + (int) below) ; 





} 
} /* Output: 
(int)above: @ 
(int)below: © 
(int)fabove: 6 
(int)fbelow: 9 
Whim 


因此 答案 是 在 将 foat 或 double 转 型 为 整 型 值 时 ， 总 是 对 该 数字 执行 截 尾 。 如 果 想 要 得 到 含 
人 的 结果 ， 就 需要 使 用 java.lang.Math 中 的 round0 方 法 : 


41: operators/RoundingNumbers .java 
// Rounding floats and doubles. 
import static net.mindview.util.Print.*; 


public class RoundingNumbers { 
public static void main(String{] args) { 








120| 
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double above = @.7, below = 0.4; 

float fabove = @.7f, fbelow = @.4f; 
print("Math.round(above): ”+ Math.round(above)): 
print ("Math. round (belo + Math.round(below)): 
print("Math.round(fabove): " + Math.round(fabove)); 
print("Math.round(fbelow): ”+ Math.round(fbelow)): 





} 
} /* Output: 
Math. round (above): 1 
Math. round (below): © 
Math.round(fabove): 1 
Math.round(fbelow): © 
“Wx 


由 于 roundO 是 java.lang 的 一 部 分 ， 因 此 在 使 用 它 时 不 需要 额外 地 导入 。 


3.15.2 提升 

如 果 对 基本 数据 类 型 执行 算术 运算 或 按 位 运算 ， 大 家 会 发 现 ， 只 要 类 型 比 int 小 ( 即 char、 
byte 或 者 short) ， 那 么 在 运算 之 前 ， 这 些 值 会 自动 转换 成 int。 这 样 一 来 ， 最 终生 成 的 结果 就 是 
int 类 型 。 如 果 想 把 结果 赋值 给 较 小 的 类 型 ， 就 必须 使 用 类 型 转换 (既然 把 结果 赋 给 了 较 小 的 类 
型 ， 就 可 能 出 现 信息 丢失 )。 通 常 ， 表 达 式 中 出 现 的 最 大 的 数据 类 型 决定 了 表达 式 最 终结 果 的 数 
据 类 型 。 如 果 将 一 个 float 值 与 一 个 double 值 相 乘 ， 结 果 就 是 double， 如 果 将 一 个 int 和 一 个 long 值 
相 加 ， 则 结果 为 long。 


3.16 Java 没 有 sizeof 


在 C 和 C++ 中 ，sizeof0 操 作 符 可 以 告诉 你 为 数据 项 分 配 的 字 节 数 。 在 C 和 C++ 中 ， 需 要 使 用 
sizeof0 的 最 大 原因 是 为 了 “移植 "。 不 同 的 数据 类 型 在 不 同 的 机 器 上 可 能 有 不 同 的 大 小 ， 所 以 在 
进行 一 些 与 存储 空间 有 关 的 运算 时 ， 程 序 员 必 须 获悉 那些 类 型 具体 有 多 大 。 例 如 ， 一 台 计 算 机 
可 用 32 位 来 保存 整数 ， 而 另 一 台 只 用 16 位 保存 。 显 然 ， 在 第 一 台 机 器 中 ， 程序 可 保存 更 大 的 值 
可 以 想像 ， 移 植 是 令 C 和 C++ 程 序 员 颇 为 头痛 的 一 个 问题 。 

Java 不 需要 sizeof0 操 作 符 来 满足 这 方面 的 需要 ， 因 为 所 有 数据 类 型 在 所 有 机 器 中 的 大 小 都 
是 相同 的 。 我 们 不 必 考 虚 移植 问题 一 它 已 经 被 设计 在 语言 中 了 。 


3.17 操作 符 小 结 


下 面 这 个 例子 向 大 家 展示 了 哪些 基本 数据 类 型 能 进行 哪些 特定 的 运算 。 基 本 上 这 是 同一 个 
不 断 重复 的 程序 ， 只 是 每 次 使 用 了 不 同 的 基本 数据 类 型 。 文 件 编译 时 不 会 报错 ， 因 为 那些 会 导 
致 编译 失败 的 行 已 用 //! 注 释 掉 了 。 


//: operators/All0ps. java 
// Tests all the operators on all the primitive data types 
// to show which ones are accepted by the Java compiler. 


public class AllOps { 
// To accept the results of a boolean test: 
void f(boolean b) {} 
void boolTest(boolean x, boolean y) { 
// Arithmetic operators: 
WLxex ty; 
Lx ex ys 
WILX EX BY; 
ML xexty; 
WV xex- y; 
INL xt; 
TNL x=; 


eoan e eS 
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II x = +y; 
IL x = -y; 
// Relational and logical: 
WE Fx > y); 

11) F(x >= y); 

I f(x < y); 

TIN f(x <= y); 

f(x == y 

fx 1= y); 

fly): 

xx BY; 

x=x Ilys 














us x << 1; 
wn x >> 1: 
WN x = x >>> 1; 
// Compound assignment: 
I x te y; 
mu 
Ws 
wt 
mt 
wt 
mn 
ms 
x & y: 
ty 
že y: 
// Casting: 
JIN char c = (char)x; 
7/1 byte b = (byte)x; 
1/1! short s = (short)x: 
J) int i = (inta; 
1/! ong 1 = (long)x; 
JAL float f = (float)x; 
J/ double d = (double)x; 
} 
void charTest(char x, char y) { 
// Arithmetic operators: 
(char) (x 
(char) (x 
(char) (x 
(char) (x 
(char) (x 





x 
x 
x 
x 


xxxxxxxx 











x = (char)+y; 
x = (char)-y; 124 
// Relational and logical: 

F(x > y); 

f(x >= y); 

f(x < y) 
f(x <= y 
f(x == y 














f(x t= y 

TIL fx); 
17) f(x 8 
WL f(x AT YDS 


// Bitwise operators: 
(char)~y; 

x = (char) (x & y); 

x = (char) (x | y); 
x = (char)(x ^ y); 
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x = (char) (x << 1); 
x = (char) (x >> 1 
x = (char) (x >>> 1); 
// Compound assignment: 
$ 5 
x 
x 
x 
x 
x 
x 
x 
x 
x 








x |= y; 
1/ Castin 
//! boolean bt = 
byte b = (byte)x 
short s = (short)x; 
int i = Cint)x; 

long 1 = (long)x; 
float f = (float)x; 
double d = (double)x; 





(boolean)x; 








} 
void byteTest(byte x, byte y) { 
// Arithmetic operators: 
x 


x 
x 
x 
x 
x 
x 


(byte) (x* y); 
(byte) (x / y); 
(byte) (x % y); 
(byte) (x + y); 
(byte) (x - y); 








x = (byte)+ y; 

x = (byte)- y; 

// Relational and logical: 
f(x > y); 

f(x >= y); 

f(x < y); 

f(x <= y); 

f(x == y); 

f(x t= y); 

WM! tax); 

I! f(x B& y); 

IA FOC 1 

// Bitwise operators: 

x = (byte)~} 
x = (byte) (x 
x = (byte) (x 
x = (byte) (x 
x = (byte) (x 
x = (byte) (x 
x = (byte) (x >>> 1) 

// Compound assignment: 
xt yt 

yi 











xxxxxxxxxx 





11 Casting: 
//! boolean bl = (boolean)x; 
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char c = (char)x; 
short s = (short)x; 
int i = (int)x; 

long 1 = (long)x; 
float f = (float)x; 
double d = (double)x; 


} 
void shortTest(short x, short y) 
// Arithmetic operators: 
x = (short) (x * y); 
x = (short)(x / y); 
x = (short) (x % y); 
x = (short) (x + y); 
x = (short)(x - 








x = (short)-y; 
// Relational and logical: 
f(x 
f(x 
tx 
f(x 
f(x 
f(x ! 
mw 
mm 
1! $ 
// Bitwise operators: 
(short)~y; 
(short) (x 
(short) (x 
(short) (x 
(short) (x 
(short) (x 
(short) (x 
7 Compound assignment: 
“y; 





nuun 











/1 Casting: 
//! boolean bl = (boolean)x; 
char c = (char)x; 
(byte)x; 
Cint)x; 
(long)x; 
float f = (float)x; 
double d = (double)x; 
} 
void intTest(int x, int y) { 
/1/ Arithmetic operators: 
xt ys 
tyi; 
% y: 
r 





y: 
y: 


xxxxxxx 
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// Relational and logical: 


f(x 
f(x 
f(x 
f(x 
f(x 





f(x! 





aa 


> yd: 
>= y); 
<y); 
<» y); 








f(x && 


y): 


J f(x 11 y): 


// Bitwise operators: 


x 
x 
x 
x 
x 
x 
/ 
x 
x -= 
x 
x 
x 
x 
x 
x 
x 
x 


WG 


7/1 boolean bl = (boolean)x; 
char c = (char)x; 

byte b = (byte)x; 

short s = (short)x; 

long 1 = (long)x: 

float f = (float)x; 
(double) x; 


doub: 
} 


void longTest(long x, long y) { 
rithmetic operators: 


LA Ai 





f(x 
f(x 
f(x 
f(x 
f(x 
f(x 
mt 
ms 
wm 


// Bitwise operators: 


x= 


xxxx 








y: 


asting: 





le d 


x 





x 
x 
x 
x 


>y); 
>= y); 
<y); 
<= y); 
== y); 
t= y); 
(1x); 
f(x Bk 
tO | 


~y: 
x &y: 
x ly: 
X ys 
x<ed; 


>>> i; 
/ Compound assignment: 
ty: 


y): 
y); 
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x 


žy 
11 Casting: 
//! boolean bl = (boolean)x; 
char ¢ = (char)x; 
byte b = (byte)x; 
short s = (short)x; 
int i = Cint)x; 
float f = (float)x; 
double d = (double)x; 

} 

void floatTest(float x, float y) { 
// Arithmetic operators: 

y: 

yi 

y: 

yi 

yi 





x ys 

// Relational and logical: 
F(x > y); 

f(x >= y); 

f(x < y); 

f(x <= y); 

f(x == y); 

f(x l= y); 

TL FC); 

JTL f(x && y); 

WY FO TEs 

// Bitwise operators: 

WIL x = mys 

ut 
In 
mn 
mh 
ms 
mt 
/1/ Compound assignment: 
x+y: 





xxxxxx 








//! boolean bl = (boolean)x; 
char c = (char)x; 
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byte b = (byte)x; 
short s = (short)x: 
int i Cint)x; 

long 1 = (long)x; 
double d = (double)x; 





void doubleTest(double x, double y) { 
// Arithmetic operators: 
x=x*y; 





x= 





// Relational and logical: 
f(x > y); 
f(x >= y); 
f(x < y); 
f(x <= y); 
f(x == y); 
f(x l= y); 
131 WA FOX): 
ITY (x && y); 
M40 1 9); 
// Bitwise operators: 
AM ~y; 
IX =X ky; 
Mx 
Mx 
Mt x 
Ix 
mtx 
// Compound assignment: 
x +s y; 
tay 
pee 





VY x l= ys 
// Casting: 
//! boolean bl = (boolean)x; 
char c = (char)x; 
byte b = (byte)x; 
short s = (short)x: 
int i = (int)x; 
long 1 = (long)x; 
float f = (float)x; 
} "A 
注意 ， 能 够 对 布尔 型 值 进行 的 运算 非常 有 限 。 我 们 只 能 赋予 它 true 和 false 值 ， 并 测试 它 为 真 
还 是 为 假 ， 而 不 能 将 布尔 值 相 加 ， 或 对 布尔 值 进行 其 他 任何 运算 。 
在 char、byte 和 short 中 ， 我 们 可 看 到 使 用 算术 操作 符 中 数据 类 型 提升 的 效果 。 对 这 些 类 型 
的 任何 一 个 进行 算术 运算 ， 都 会 获得 一 个 int 结 果 ， 必 须 将 其 显 式 地 类 型 转换 回 原来 的 类 型 ( 罕 
化 转换 可 能 会 造成 信息 的 丢失 ) ， 以 将 值 赋 给 原本 的 类 型 。 但 对 于 int 值 ， 却 不 必 进 行 类 型 转化 ， 
因为 所 有 数据 都 已 经 属于 int 类 型 。 但 不 要 放松 警惕 ， 认 为 一 切 事 情 都 是 安全 的 ， 如 果 对 两 个 足 


pe ee 
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够 大 的 int 值 执行 乘法 运算 ， 结 果 就 会 溢出 。 下 面 这 个 例子 向 大 家 展示 了 这 一 点 : 


//: operators/Overflow. java 
// Surprise! Java lets you overflow. 


public class Overflow { 
public static void main(String[] args) { 
int big = Integer .MAX_VALUE; 
System.out.printin("big = ”+ big); 
int bigger = big * 4; 
System.out.printin("bigger = ”+ bigger): 


} 
} /* Output: 
big = 2147483647 
bigger = -4 
Whi 


你 不 会 从 编译 器 那里 收 到 出 错 或 警告 信息 ， 运 行 时 也 不 会 出 现 异常 。 这 说 明 Java 虽 然 是 好 


东西 ， 但 也 没有 那么 好 ! 

对 于 char、byte 或 者 short， 复 合 赋值 并 不 需要 类 型 转换 。 尽 管 它们 执行 类 型 提升 ， 但 也 会 
获得 与 直接 算术 运算 相同 的 结果 。 而 在 另 一 方面 ， 省 略 类 型 转换 可 使 代码 更 加 简练 。 

可 以 看 到 ， 除 boolean 以 外 ， 任 何 一 种 基本 类 型 都 可 通过 类 型 转换 变 为 其 他 基本 类 型 。 再 一 
次 提醒 读者 。 当 类 型 转换 成 一 种 较 小 的 类 型 时 ， 必 须 留意 “ 窗 化 转换 ”的 结果 ， 否 则 会 在 类 型 
转化 过 程 中 不 知 不 觉 地 丢失 了 信息 。 

练习 14，(3) 编写 一 个 接收 两 个 字符 串 参数 的 方法 ， 用 各 种 布尔 值 的 比较 关系 来 比较 这 两 个 
字符 串 ， 然 后 把 结果 打印 出 来 。 做 == 和 != 比较 的 同时 ， 用 equals0 作 测试 。 在 main0 里 面 用 几 个 
不 同 的 字符 串 对 象 调用 这 个 方法 。 


3.18 总 结 


如 果 你 拥有 编程 语言 的 经 验 ， 那 么 只 要 它 的 语法 类 似 于 C， 你 就 会 发 现 Java 中 的 操作 符 与 它 
们 是 多 么 地 相似 ， 以 至 于 对 你 来 说 没有 任何 学 习 困难 。 如 果 你 发 现 本 章 颇具 挑战 性 ， 那 么 你 应 
该 先 阅读 从 www.MindView.net 上 可 获得 的 Thinking in C 多 媒体 材料 。 

所 选 习题 的 答案 都 可 以 在 名 为 The Thinking in Java Annotated Solution Guide 的 电子 文档 中 找 
到 ， 读 者 可 以 从 www.MindView.net 上 购买 此 文档 。 
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第 4 章 控制 执行 流程 

就 像 有 知觉 的 生物 一 样 ， 程 序 必 须 在 执行 过 程 中 控制 它 的 世界 ， 并 做 出 选择 。 在 Java 中 ， 
你 要 使 用 执行 控制 语句 来 做 出 选择 。 

Java 使 用 了 C 的 所 有 流程 控制 语句 ， 所 以 如 果 读 者 以 前 用 过 C 或 C++ 编程 ， 那 么 应 该 非常 熟 
悉 了 。 大 多 数 过 程 型 编程 语言 都 具有 某 些 形式 的 控制 语句 ， 它 们 通常 在 各 种 语言 间 是 交 迭 的 。 
在 Java 中 ， 涉 及 的 关键 字 包 括 if-else、while、do-while、for、return、break 以 及 选择 语句 switch。 
然而 ，Java 并 不 支持 goto 语 句 (该 语句 引起 许多 反对 意见 ， 但 它 仍 是 解决 某 些 特殊 问题 的 最 便利 
的 方法 ) 。 在 Java 中 ， 仍 然 可 以 进行 类 似 goto 那 样 的 跳 转 ， 但 比 起 典型 的 goto， 有 了 很 多 限制 。 


4.1 true 和 false 


所 有 条 件 语句 都 利用 条 件 表达 式 的 真 或 假 来 决定 执行 路 径 。 这 里 有 一 个 条 件 表 达 式 的 例子 ， 
a==b。 它 用 条 件 操作 符 “ 一 ”来 判断 a 值 是 否 等 于 b 值 。 该 表达 式 返 回 true 或 false。' 本 章 前 面 介 
绍 的 所 有 关系 操作 符 ， 都 可 拿 来 构造 条 件 语句 。 注 意 Java 不 克 许 我 们 将 一 个 数字 作为 布尔 值 使 
用 ， 虽 然 这 在 C 和 C++ 里 是 允许 的 〈 在 这 些 语言 里 ,“ 真 ”是 非 零 ， 而 “ 假 ”是 零 )。 如 果 想 在 布 
尔 测试 中 使 用 一 个 非 布尔 值 ， 比 如 在 if(a) 中 ， 那 么 首先 必须 用 一 个 条 件 表达 式 将 其 转换 成 布尔 
值 ， 例 如 if(a!=0)。 


4.2 if-else 


if-else 语 句 是 控制 程序 流程 的 最 基本 的 形式 。 基 中 的 else 是 可 选 的 ， 所 以 可 按 下 述 两 种 形式 
来 使 用 过: 


if (Boolean-expression) 
statement 


或 


1f(Bootean-expression) 
Statement 

else 
statement 


布尔 表达 式 必 须 产生 一 个 布尔 结果 ，statement 指 用 分 号 结尾 的 简单 语句 ， 或 复合 语句 一 _ 封 
闲 在 花 括 号 内 的 一 组 简单 语句 。 在 本 书 任何 地 方 ， 只 要 提 及 “语句 ”这 个 词 ， 就 指 的 是 简单 语 
名 或 复合 语句。 

作为 记 else 的 一 个 例子 ， 下 面 这 个 test0 方 法 可 以 告诉 您 ， 您 猜 的 数 是 大 于 、 小 于 还 是 等 于 目 
标 数 ， 


//: control/IfElse. java 
import static net.mindview.util.Print.*; 


public class IfElse { 
static int result = 0; 
static void test(int testval, int target) { 
if(testval > target) 
result = +1; 
else if(testval < target) 
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result = -1; 
else 
result = 0; // Match 
bpue static void main(String[] args) { 
test(10, 5); 
print(result); 
test(5, 10); 
print(result); 
test(5, 5); 
print(result); 
} 六 Output: 
1 
-1 
9 
Wh 
在 test0 的 中 间 部 分 ， 可 以 看 到 一 个 “else if ， 那 并 非 新 的 关键 字 ， 而 仅仅 只 是 一 个 else 后 面 
紧 跟 另 一 个 新 的 这 语句 。 
尽管 Java 与 它 之 前 产生 的 C 和 C++ 一 样 ， 都 是 “格式 自由 ”的 语言 ， 但 是 习惯 上 还 是 将 流程 
控制 语句 的 主体 部 分 缩 进 排列 ， 使 读者 能 方便 地 确定 起 始 与 终止 。 


4.3 迭代 


while、do-while 和 for 用 来 控制 循环 ， 有 时 将 它们 划分 为 选 代 语句 (iteration statement)。 语 
名 会 重复 执行 ， 直 到 起 控制 作用 的 布尔 表达 式 (Booleanexpression) 得 到 “ 假 ” 的 结果 为 止 。 
while 循 环 的 格式 如 下 : 


while(Boolean-expression) 
statement 


在 循环 刚 开始 时 ， 会 计算 一 次 布尔 表达 式 的 值 ， 而 在 语句 的 下 一 次 迭代 开始 前 会 再 计算 
一 次 。 
下 面 这 个 简单 的 例子 可 产生 随机 数 ， 直 到 符合 特定 的 条 件 为 止 : 


//: control/whileTest.java 
// Demonstrates the while loop. 


public class WhileTest { 
static boolean condition() { 
boolean result = Math.random() < 0.99; 
System.out.print(result +", "); 
return result; 


} 
public static void main(String[} args) { 
whi le(condition()) 
System.out.printin("Inside ‘while'"); 
System.out.printin("Exited 'while'"); 


} 
} /* (Execute to see output) *///:~ 


condition0 方 法 用 到 了 Math 库 里 的 statie (静态 ) 方法 random()， 该 方法 的 作用 是 产生 0 和 
1 之 间 (包括 9， 但 不 包括 1) 的 一 个 double 值 。result 的 值 是 通过 比较 操作 符 < 而 得 到 它 ， 这 个 
操作 符 将 产生 boolean 类 型 的 结果 。 在 打印 boolean 类 型 的 值 时 ， 将 自动 地 得 到 适合 的 字符 串 
tme 或 false。while 的 条 件 表达 式 意思 是 说 :“ 只 要 condition0 返 回 true， 就 重复 执行 循环 体 中 的 
语句 ”。 
4.3.1 do-while 

do-while 的 格式 如 下 : 
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do 
statement 
whi le(Boolean-expression); 


while 和 do-while 唯 一 的 区 别 就 是 do-while 中 的 语句 至 少 会 执行 一 次 ， 即 便 表达 式 第 一 次 就 被 
计算 为 false。 而 在 while 循 环 结构 中 ， 如 果 条 件 第 一 次 就 为 false， 那 么 其 中 的 语句 根本 不 会 执行 。 
在 实际 应 用 中 ，while 比 do-while 更 常用 一 些 。 

4.3.2 for 

for 循 环 可 能 是 最 经 常 使 用 的 和 迭代 形式 ， 这 种 在 第 一 次 和 迭代 之 前 要 进行 初始 化 。 随 后 ， 它 会 

进行 条 件 测试 ， 而 且 在 每 一 次 迭代 结束 时 ， 进 行 某 种 形式 的 “ 步 进 "。for 循 环 的 格式 如 下 : 


for(initialization; Boolean-expression; step) 
statement 


初始 化 (initialization) 表达 式 、 布 尔 表达 式 (Boolean-expression) ， 或 者 步 进 (step) 运算 ， 
都 可 以 为 空 。 每 次 选 代 前 会 测试 布尔 表达 式 。 若 获得 的 结果 是 false， 就 会 执行 for 语 句 后 面 的 代 
码 行 。 每 次 循环 结束 ， 会 执行 一 次 步 进 。 

for 循 环 常用 于 执行 “计数 ”任务 : 


/1: control/ListCharacters.java 
// Demonstrates "for" loop by listing 
11 all the lowercase ASCII letters. 


public class ListCharacters { 
public static void main(String[] args) { 
for(char c = 0; c < 128; c++) 
if (Character. isLowerCase(c)) 
System.out.printIn(*value: " + (int)c + 
"character: " + c); 


} 

} /* Output: 

value: 97 character: a 
value: 98 character: b 
value: 99 character: < 
value: 100 character: 

value: 101 character: 

value: 102 character: 

value: 103 character: 

value: 104 character: 

value: 105 character: 

value: 106 character: 


一 一 了 mm oo 


s~ 

注意 ， 变 量 是 在 程序 用 到 它 的 地 方 被 定义 的 ， 也 就 是 在 for 循 环 的 控制 表达 式 里 ， 而 不 是 在 
main0 开 始 的 地 方 定 义 的 。*e 的 作用 域 就 是 for 控 制 的 表达 式 的 范围 内 。 

这 个 程序 也 使 用 了 javalang.Character 包 装 器 类 ， 这 个 类 不 但 能 把 char 基 本 类 型 的 值 包装 进 
对 象 ， 还 提供 了 一 些 别 的 有 用 的 方法 。 这 里 用 到 了 static isLowerCase() 方 法 来 检查 问题 中 的 字符 
是 否 为 小 写字 母 。 

对 于 像 C 语 言 那样 的 传统 的 过 程 型 语言 ， 要 求 所 有 变量 都 在 一 个 块 的 开头 定义 ， 以 便 编译 器 
在 创建 这 个 块 的 时 候 ， 可 以 为 那些 变量 分 配 空间 。 而 在 Java 和 C++ 中 ， 则 可 在 整个 块 的 范围 内 分 
散 变量 声明 ， 在 真正 需要 的 地 方才 加 以 定义 。 这 样 便 可 形成 更 自然 的 编程 风格 ， 也 更 易 理 解 。 

练习 1:(1) 写 一 个 程序 ， 打 印 从 1 到 100 的 值 。 

练习 2: (2) 写 一 个 程序 ， 产 生 25 个 int 类 型 的 随机 数 。 对 于 每 一 个 随机 值 ， 使 用 if-else 语 句 来 
将 其 分 类 为 大 于 、 小 于 ,或 等 于 紧 随 它 而 随机 生成 的 值 。 
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练习 3，(1) 修改 练习 2， 把 代码 用 一 个 while 无 限 循环 包括 起 来 。 然 后 运行 它 直 至 用 键盘 中 断 
其 运行 (通常 是 通过 按 Ctrl-C)。 

练习 4: (3) 写 一 个 程序 ， 使 用 两 个 嵌 套 的 for 循 环 和 取 余 操作 符 (%) 来 探测 和 打印 素数 
(只 能 被 其 自身 和 1 整除 ， 而 不 能 被 其 他 数字 整除 的 整数 ) 。 

S35. (4) 重复 第 3 章 中 的 练习 10， 不 要 用 Integer.toBinaryString0 方 法 ， 而 是 用 三 元 操作 
符 和 按 位 操作 符 来 显示 二 进 制 的 1 和 0。 s 
43.3 逗号 操作 符 

本 章 前 面 已 经 提 到 了 逗号 操作 符 (注意 不 是 逗号 分 隔 符 ， 喜 号 用 作 分 隔 符 时 用 来 分 隔 函 数 
的 不 同 参数 ) ，Java 里 唯一 用 到 逗号 操作 符 的 地 方 就 是 for 循 环 的 控制 表达 式 。 在 控制 表达 式 的 初 
始 化 和 步 进 控制 部 分 ， 可 以 使 用 一 系列 由 逗号 分 隔 的 语句 ， 而 且 那 些 语句 均 会 独立 执行 。 

通过 使 用 逗号 操作 符 ， 可 以 在 for 语 句 内 定义 多 个 变量 ， 但 是 它们 必须 具有 相同 的 类 型 。 


//: controt/Comma0perator .java 


public class Comma0perator { 
public static void main(String[] args) { 
for(int 1 = 1, j= i + 10; 1 < 5; i++, fj = 4% 2) { 
System.out.println("i = "+4+" j=" + 4); 


utpu 
j= 
j= 
j= 

ie 

for 语 句 中 的 int 定 义 涵盖 了 i 和 ij， 在 初始 化 部 分 实际 上 可 以 拥有 任意 数量 的 具有 相同 类 型 的 
变量 定义 。 在 一 个 控制 表达 式 中 ， 定 义 多 个 变量 的 这 种 能 力 只 限于 for 循 环 适用 ， 在 其 他 任何 选 
择 或 迭代 语句 中 都 不 能 使 用 这 种 方式 。 

可 以 看 到 ， 无 论 在 初始 化 还 是 在 步 进 部 分 ， 语 句 都 是 顺序 执行 的 。 此 外 ， 初 始 化 部 分 可 以 
有 任意 数量 的 同一 类 型 的 定义 。 


4.4 Foreach 语 法 


Java SE5 引 入 了 一 种 新 的 更 加 简洁 的 for 语 法 用 于 数组 和 容器 (在 第 16 章 与 第 17 章 中 将 更 多 
地 讨论 这 种 语法 ) ， 即 foreach 语 法 ， 表 示 不 必 创建 int 变 量 去 对 由 访问 项 构成 的 序列 进行 计数 ， 
foreach 将 自动 产生 每 一 项 。 

例如 ， 假 设 有 一 个 float 数 组 ， 我 们 要 选取 该 数组 中 的 每 一 个 元 素 : 


//: control/ForEachFloat. java 
‘import java.util.*; 


public class ForEachFloat { 
public static void main(String(] args) { 
Random rand = new Random(47); 
float f{] = new float[10]; 
for(int i = @; i < 16; i++) 
f[1] = rand.nextFloat(): 
for(float x : f) 
System. out.printin(x) ; 


} 
} /* Output: 
6.72711575 
0.39982635 
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0.5309454 
@.0534122 
@.16026656 
6.57799757 
8.18847865 
8.4179137 
0.51660204 
6.73734957 
Whim 


这 个 数组 是 用 旧式 的 for 循 环 组 装 的 ， 因 为 在 组 装 时 必须 按 索 引 访问 它 。 在 下 面 这 行 中 可 以 
看 到 foreach 语 法 ; 

for(float x : f) 《 

这 条 语句 定义 了 一 个 foat 类 型 的 变量 x， 继 而 将 每 一 个 f 的 元 素 赋值 给 x。 

任何 返回 一 个 数组 的 方法 都 可 以 使 用 foreach。 例 如 ，String 类 有 一 个 方法 toCharArray0， 
它 返 回 一 个 char 数 组 ， 因 此 可 以 很 容易 地 像 下 面 这 样 选 代 在 字符 串 里 面 的 所 有 字符 ; 
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public class ForEachString { 
public static void main(String[] args) { 
for(char c : "An African Swallow". toCharArray() ) 
System.out.print(c + " "); 


} 

} /* Output: 

An African Swallow 
Whim 


就 像 在 第 11 章 中 所 看 到 的 ，foreach 还 可 以 用 于 任何 Iterable 对 象 。 

许多 for 语 句 都 会 在 一 个 整 型 值 序列 中 步 进 ， 就 像 下 面 这 样 : 

for(int i = 0; 1 < 100; i++) 

对 于 这 些 语句 ，foreach 语 法 将 不 起 作用 ， 除 非 先 创建 一 个 int 数 组 。 为 了 简化 这 些 任务 ， 我 
在 net.mindview.util.Range 包 中 创建 了 一 个 名 为 range0 的 方法 ， 它 可 以 自动 地 生成 恰当 的 数组 。 
我 的 目的 是 将 range0 用 做 static 导 入: 


//: controt/ForEachInt .java 
import static net.mindview.util.Range.*; 
import static net.mindview.util.Print.*; 


public class ForEachint { 
public static void main(String{] args) { 

for(int i : range(1®)) // 6..9 
printnb(i +" "); 

print(); 

for(int i : range(5, 16)) // 5..9 
printnb(i +" "); 

print(); 

for(int i : range(5, 26, 3)) // 5..20 step 3 
printnb(i +" "); 

print(); 
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range( 方 法 已 经 被 重 载 ， 重 载 表示 相同 的 方法 名 可 以 具有 不 同 的 参数 列表 (你 将 很 快 学 习 
重 载 )。range0 的 第 一 种 重 载 形式 是 从 0 开始 产生 值 ， 直 至 范围 的 上 限 ， 但 不 包括 该 上 限 。 第 二 
种 形式 从 第 一 个 值 开 始 产生 值 ， 直 至 比 第 二 个 值 小 1 的 值 为 止 。 第 三 种 形式 有 一 个 步 进 值 ， 因 此 
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它 每 次 的 增 量 为 该 值 。range0 是 所 谓 生 成 器 的 一 个 非常 简单 的 版 本 ， 有 关 生 成 器 的 内 容 将 在 本 
书 稍 后 进行 介绍 。 

请 注意 ， 尽 管 range0 使 得 foreach 语 法 可 以 适用 于 更 多 的 场合 ， 并 且 这 样 做 似乎 可 以 增加 可 
读 性 ， 但 是 它 的 效率 会 稍 许 降 低 ， 因 此 如 果 您 在 做 性 能 调 优 ， 也 许 应 该 使 用 仿真 器 来 做 评价 ， 
它 是 一 种 可 以 度量 代码 性 能 的 工具 。 

你 会 注意 到 ， 除 了 print0 之 外 ， 我 们 还 使 用 了 printnbO0。printnb0 方 法 不 会 换行 ， 因 此 可 以 
使 用 它 将 一 行 拆 分 成 多 个 片断 输出 。 

foreach 语 法 不 仅 在 录入 代码 时 可 以 节省 时 间 ， 更 重要 的 是 ， 它 阅读 起 来 也 要 容易 得 多 ， 它 
说 明 您 正在 努力 做 什么 〈 例 如 获取 数组 中 的 每 一 个 元 素 ) ， 而 不 是 给 出 你 正在 如 何 做 的 细节 (A 
如 正在 创建 索引 ， 因 此 可 以 使 用 它 来 选取 数组 中 的 每 一 个 元 素 ) 。 在 本 书 中 ， 我 们 只 要 有 可 能 就 
会 使 用 foreach 语 法 。 


4.5 return 


在 Java 中 有 多 个 关键 词 表示 无 条 件 分 支 ， 它 们 只 是 表示 这 个 分 支 无 需 任何 测试 即 可 发 生 。 
这 些 关键 词 包括 return、break、continue 和 一 种 与 其 他 语言 中 的 goto 类 似 的 跳 转 到 标号 语句 的 
方式 。 

retum 关 键 词 有 两 方面 的 用 途 : 一 方面 指定 一 个 方法 返回 什么 值 (假设 它 没有 void 返回 值 )， 另 
一 方面 它 会 导致 当前 的 方法 退出 ， 并 返回 那个 值 。 可 据 此 改写 上 面 的 test0 方 法 ， 使 其 利用 这 些 特点 : 

/1/: controt/IfELse2.java 

import static net.mindview.util.Print.*; 


public class IfElse2 { 
static int test(int testval, int target) { 
if(testval > target) 
return +1; 
else if(testval < target) 
return -1; 
else 
return 6; // Match 


} 
public static void main(String{] args) { 
print(test(10, 5)); 
print(test(5, 10)); 
print(test(5, 5)): 
} 
} /* Output: 
1 
-1 
9 
Wha 


不 必 加 上 else， 因 为 方法 在 执行 了 return 后 不 再 继续 执行 。 

如 果 在 返回 void 的 方法 中 没有 return 语 句 ， 那 么 在 该 方法 的 结尾 处 会 有 一 个 隐 式 的 return, 
因此 在 方法 中 并 非 总 是 必须 要 有 一 个 retum 语 句 。 但 是 ， 如 果 一 个 方法 声明 它 将 返回 void 之 外 的 
其 他 东西 ， 那 么 必须 确保 每 一 条 代码 路 径 都 将 返回 一 个 值 。 

练习 6: (2) 修改 前 两 个 程序 中 的 两 个 test0 方 法 ， 让 它们 接受 两 个 额外 的 参数 begin 和 end ， 
这 样 在 测试 testval 时 将 判断 它 是 否 在 begin 和 end 之 间 (包括 begin 和 end) 的 范围 内 。 


4.6 break 和 continue 
在 任何 迭代 语句 的 主体 部 分 ， 都 可 用 break 和 continue 控 制 循环 的 流程 。 其 中 ，break 用 于 强 
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行 退 出 循环 ， 不 执行 循环 中 剩余 的 语句 。 而 continue 则 停止 执行 当前 的 迭代 ， 然 后 退回 循环 起 始 
处 ， 开 始 下 一 次 迭代 。 
下 面 这 个 程序 向 大 家 展示 了 break 和 continue 在 for 和 while 循 环 中 的 例子 : 


//: control/BreakAndContinue. java 

// Demonstrates break and continue keywords. 

import static net.mindview.util.Range.*; 

public class BreakAndContinue { 

public static void main(String] args) { 
for(int 1 = @; i < 100; i++) { 

if(i == 74) break: // Out of for loop 
if(i % 9 != 8) continue; // Next iteration 
System.out.print(i + " "); 





} 

System.out.printin(); 

71 Using foreach: 

for(int i : range(199)) { 
1f(i == 74) break; // Out of for loop 
if(i % 9 != 0) continue; // Next iteration 
System.out.print(i +" "); 

} 

System.out.printin(); 

int 1 = @: 

// An “infinite loop": 

while(true) { 
itt; 
int j = 1 * 27; 
if(j == 1269) break; // Out of loop 
if(i % 10 != @) continue; // Top of loop 
System.out.print(i +" "); 

} 


} 

} /* Output: 

9 9 18 27 36 45 54 63 72 
@ 9 18 27 36 45 54 63 72 
10 20 30 40 

“N~ 


在 这 个 for 循 环 中 ，i 的 值 永远 不 会 达到 100， 因 为 一 旦 ji 到达 74，break 语 名 就 会 中 断 循环 。 
通常 ， 只 有 在 不 知道 中 断 条 件 何 时 满足 时 ， 才 需要 这 样 使 用 break。 只 要 i 不 能 被 9 整除 ， 
continue 语 句 就 会 使 执行 过 程 返 回 到 循环 的 最 开头 (这 使 值 递增 ) 。 如 果 能 够 整除 ， 则 将 值 显示 
出 来 。 

第 二 种 for 循 环 展示 了 foreach 用 法 ， 它 将 产生 相同 的 结果 。 

最 后 ， 可 以 看 到 一 个 “无 穷 While 循 环 ” 的 情况 。 然 而 ， 循 环 内 部 有 一 个 break 语 句 ， 可 中 止 
循环 。 除 此 以 外 ， 大 家 还 会 看 到 continue 语 句 执行 序列 移 回 到 循环 的 开头 ， 而 没有 去 完成 
continue 语 名 之 后 的 所 有 内 容 。( 只 有 在 i 值 能 被 10 整 除 时 才 打印 出 值 。) 输出 结果 之 所 以 显示 0， 


是 由 于 0%9 等 于 0。 
无 穷 循环 的 第 二 种 形式 是 for(;;)。 编 译 器 将 while(true) 与 for(;;) 看 作 是 同一 回 事 。 所 以 具体 
选用 哪个 取决 于 自己 的 编程 习惯 。 


练习 7: (1) 修改 本 章 练习 1， 通 过 使 用 break 关 键 词 ， 使 得 程序 在 打印 到 99 时 退出 。 然 后 举 
试 使 用 return 来 达到 相同 的 目的 。 


4.7 臭名 昭著 的 goto 


编程 语言 中 一 开始 就 有 goto 关 键 词 了 。 事 实 上 ，goto 起 源 于 汇编 语言 的 程序 控制 :“ 若 条 件 
A 成 立 ， 则 跳 到 这 里 ， 否 则 跳 到 那里 "。 如 果 阅 读 由 编译 器 最 终生 成 的 汇编 代码 ， 就 会 发 现 程序 


_—— 
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控制 里 包含 了 许多 跳 转 。(Java 编 译 器 生成 它 自己 的 “汇编 代码 ”"， 但 是 这 个 代码 是 运行 在 Java 虚 
拟 机 上 的 ， 而 不 是 直接 运行 在 CPU 硬件 上 。) 

goto 语 句 是 在 源码 级 上 的 跳 转 ， 这 使 其 招致 了 不 好 的 声誉 。 若 一 个 程序 总 是 从 一 个 地 方 跳 
到 另 一 个 地 方 ， 还 有 什么 办 法 能 识别 程序 的 控制 流程 呢 ? 自从 Edsger Dijkstra 发 表 了 著名 论文 
(Goto considered harmful》(Goto 有 害 ) ， 众 人 开始 痛斥 goto 的 不 是 ， 甚 至 建议 将 它 从 关键 词 集合 
中 扫地 出 门 。 

对 于 这 个 问题 ， 中 庸 之 道 是 最 好 解决 方法 。 真 正 的 问题 并 不 在 于 使 用 goto， 而 在 于 goto 的 滥 
用 ， 而 且 少数 情况 下 ，goto 还 是 组 织 控制 流程 的 最 佳 手段 。 

尽管 goto 仍 是 Java 中 的 一 个 保留 字 ， 但 在 语言 中 并 未 使 用 它 ，Java 没 有 goto。 然 而 ，Java 也 ， 
能 完成 一 些 类 似 于 跳 转 的 操作 ， 这 与 break 和 continue 这 两 个 关键 词 有 关 。 它 们 其 实 不 是 一 个 跳 
转 ， 而 是 中 断 迭 代 语句 的 一 种 方法 。 之 所 以 把 它们 纳入 goto 问 题 中 一 起 讨论 ， 是 由 于 它们 使 用 
了 相同 的 机 制 ， 标签 。 

标签 是 后 面 跟 有 冒号 的 标识 符 ， 就 像 下 面 这 样 : 

labell: 

在 Java 中 ， 标 签 起 作用 的 唯一 的 地 方刚 好 是 在 迭代 语句 之 前 。“ 刚 好 之 前 ”的 意思 表明 ， 在 
标签 和 选 代 之 间 置 和 任何 语句 都 不 好 。 而 在 选 代 之 前 设置 标签 的 唯一 理由 是 : 我 们 希望 在 其 中 
侈 套 另 一 个 迭代 或 者 一 个 开关 《你 很 快 就 会 学 习 到 它 )。 这 是 由 于 break 和 continue 关 键 词 通常 只 
中 断 当前 循环 ， 但 若 随同 标签 一 起 使 用 ， 它 们 就 会 中 断 循环 ， 直 到 标签 所 在 的 地 方 : 


label1: 
outer-iteration { 
inner-iteration { 
Whewe 
break; // (1) 
Mass 
continue; // (2) 
Mave 
continue tabell; // (3) 
Wane 
break labell; // (4) 
k 
H 


在 (D 中 ，break 中 断 内 部 选 代 ， 回 到 外 部 迭代 。 在 (2) 中 ，eontinue 使 执行 点 移 回 内 部 迭代 的 
起 始 处 。 在 (3) 中 ，continue labell 同 时 中 断 内 部 选 代 以 及 外 部 选 代 ， 直 接 转 到 labell 处 ， 随 后 ， 
它 实际 上 是 继续 先 代 过 程 ， 但 却 从 外 部 迭代 开始 。 在 (9 中 ，break labell 也 会 中 断 所 有 迭代， 并 
回 到 labell 处 ， 但 并 不 重新 进入 迭代。 也 就 是 说 ， 它 实际 是 完全 中 止 了 两 个 迭代 。 

下 面 是 标签 用 于 for 循 环 的 例子 : 


//: control/LabeledFor .java 
// For loops with "labeled break” and “labeled continue.” 
import static net.mindview.util.Print.*; 


public class LabeledFor { 
public static void main(String[] args) { 
int 1 = 0; 
outer: // Can't have statements here 
for(; true ;) { // infinite loop 
inner: // Can't have statements here 
for(i i < 10; i++) { 
print("i =" + i); 
if(1 == 2) { 
print("continue"); 
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continue; 


y 
if(i == 3) { 
print("break"); 
i++; // Otherwise i never 
// gets incremented. 
break; 


} 
if(i == 7) { 
print("continue outer”); 
i++; // Otherwise 1 never 
// gets incremented. 
continue outer; 


} 

if(1 == 8) { 
print("break outer"); 
break outer; 


for(int k = 9: k < 5; k++) { 
if(k == 3) { 
print("continue inner"); 
continue inner; 
} 
} 
} 





} 
// Can't break or continue to labels here 


} 
} /* Output: 
iso 
continue inner 
i=l 
continue inner 
i=? 
continue 
1=3 
break 
1=4 
continue inner 
1=5 
continue inner 
i=6 
continue inner 
1=7 
continue outer 
i=8 
break outer 
Whim 


注意 ，break 会 中 断 for 循 环 ， 而 且 在 抵达 for 循 环 的 末尾 之 前 ， 递 增 表达 式 不 会 执行 。 由 于 
break 跳 过 了 递增 表达 式 ， 所 以 在 i==3 的 情况 下 直接 对 i 执 行 递 增 运算 。 在 i==7 的 情况 下 ， 
continue outer 语 句 会 跳 到 循环 顶部 ， 而 且 也 会 跳 过 递增 ， 所 以 这 里 也 对 i 直接 递增 。 

如 果 没 有 break outer 语 句 ， 就 没有 办 法 从 内 部 循环 里 跳出 外 部 循环 。 这 是 由 于 break 本 身 只 
能 中 断 最 内 层 的 循环 (continue 同 样 也 是 如 此 ) 。 

当然 ， 如 果 想 在 中 断 循环 的 同时 退出 ， 简 单 地 用 一 个 return 即 可 。 

下 面 这 个 例子 向 大 家 展示 了 带 标签 的 break 以 及 continue 语 句 在 while 循 环 中 的 用 法 : 


//: control/Labeledwhile. java 
// While loops with “Labeled break" and “labeled continue.” 
import static net.mindview.util.Print.*; 


public class LabeledWhile { 
public static void main(String[] args) { 
int i = @; 
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outer: 
while(true) { 
print("Outer while loop"); 
while(true) { 
itt; 
print("i = * + i); 
1f(1 == 1) { 
print("continue"); 
continue; 
} 
ifG == 3) { 
print("continue outer"): 
continue outer; 





} 

if == 5) { 
print ("break"); 
break; 


} 
if(1 == 7) { 
print("break outer"); 
break outer; 
} 
} 
} 


} Output: 

Outer while loop 

ie. 

continue 

i=2 

i=3 

continue outer 

Outer while loop 

1=4 

1=5 

break 

Outer while loop 

i=6 

ia? 

break outer 

SHl :~ 

同样 的 规则 亦 适 用 于 while: 

1) 一 般 的 continue 会 退回 最 内 层 循环 的 开头 (顶部) ， 并 继续 执行 。 

2) 带 标签 的 continue 会 到 达标 签 的 位 置 ， 并 重新 进入 紧 接 在 那个 标签 后 面 的 循环 。 

3) 一 般 的 break 会 中 断 并 跳出 当前 循环 。 

4) 带 标签 的 break 会 中 断 并 跳出 标签 所 指 的 循环 。 

要 记 住 的 重点 是 : 在 Java 里 需要 使 用 标签 的 唯一 理由 就 是 因为 有 循环 候 套 存在 ， 而 且 想 从 
多 层 供 套 中 break 或 continue。 

在 Dijkstra 的 《Goto 有 害 》 的 论文 中 ， 他 最 反对 的 就 是 标签 ， 而 非 goto。 他 发 现在 一 个 程序 
里 随 着 标签 的 增多 ， 产 生 的 错误 也 会 越 来 越 多， 并 且 标签 和 goto 使 得 程序 难以 分 析 。 但 是 ，Java 
的 标签 不 会 造成 这 种 问题 ， 因 为 它们 的 应 用 场合 已 经 受到 了 限制 ， 没 有 特别 的 方式 用 于 改变 程 
序 的 控制 。 由 此 也 引出 了 一 个 有 趣 的 现象 : 通过 限制 语句 的 能 力 ， 反 而 能 使 一 项 语言 特性 更 加 
有 用 。 


4.8 switch 


Switch 有 时 也 被 划 归 为 一 种 选择 语句 。 根 据 整数 表达 式 的 值 ，switch 语 句 可 以 从 一 系列 代码 
中 选 出 一 段 去 执行 。 它 的 格式 如 下 : 
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switch(integral-selector) { 
case integral-valuel : statement; break; 
case integral-value2 : statement; break; 
case integral-value3 : statement; break; 
case integral-value4 : statement; break; 
case integral-valueS : statement; break; 
VIA 
default: statement; 

} 


其 中 ，Integral-selector (整数 选择 因子 ) 是 一 个 能 够 产生 整数 值 的 表达 式 ，switch 能 将 这 个 
表达 式 的 结果 与 每 个 integral-value (整数 值 ) 相 比较 。 若 发 现 相符 的 ， 就 执行 对 应 的 语句 (单一 
语句 或 多 条 语句 ， 其 中 并 不 需要 括号 ) 。 若 没有 发 现 相符 的 ， 就 执行 default (默认 ) 语句 。 

在 上 面 的 定义 中 ， 大 家 会 注意 到 每 个 case 均 以 一 个 break 结 尾 ， 这 样 可 使 执行 流程 跳 转 至 
switch 主 体 的 末尾 。 这 是 构建 switch 语 句 的 一 种 传统 方式 ， 但 break 是 可 选 的 。 若 省 略 break， 会 
继续 执行 后 面 的 case 语 句 ， 直 到 遇 到 一 个 break 为 止 。 尽 管 通常 不 想 出 现 这 种 情况 ， 但 对 有 经 验 
的 程序 员 来 说 ， 也 许 能 够 善 加 利用 这 种 情况 。 注 意 最 后 的 default 语 句 没有 break， 因 为 执行 流程 
已 到 了 break 的 跳 转 目的 地 。 当 然 ， 如 果 考 虑 到 编程 风格 方面 的 原因 ， 完 全 可 以 在 default 语 句 的 
末尾 放置 一 个 break， 尽 管 它 并 没有 任何 实际 的 用 处 。 

Switch 语句 是 实现 多 路 选择 (也 就 是 说 从 一 系列 执行 路 径 中 挑选 一 个 ) 的 一 种 干净 利落 的 方 
法 。 但 它 要 求 使 用 一 个 选择 因子 ， 并 且 必须 是 int 或 char 那 样 的 整数 值 。 例 如 ， 假 若 将 一 个 字符 
申 或 者 浮 点 数 作为 选择 因子 使 用 ， 那 么 它们 在 switeh 语 句 里 是 不 会 工作 的 。 对 于 非 整 数 类 型 ， 则 
必须 使 用 一 系列 这 语句 。 在 下 一 章 的 末尾 ， 你 将 看 到 Java SE5 的 新 特性 enum， 它 可 以 帮助 我 们 减 
弱 这 种 限制 ， 因 为 enum 可 以 和 switch 协 调 工作 。 

下 面 这 个 例子 可 随机 生成 字母 ， 并 判断 它们 是 元 音 还 是 辅音 字母 ; 


//: control/VowelsAndConsonants. java 

// Demonstrates the switch statement. 
import java.utit.*; 

import static net.mindview.util.Print.*; 


public class VowelsAndConsonants { 
public static void main(String[] args) { 
Random rand = new Random(47) ; 
for(int 1 = @; i < 100; i++) { 
int c = rand.nextint(26) + 'a' 
printnb((char)c +", “#e +": "); 
Switch(c) { 
case ‘a’: 
case 
case 
case ‘o': 
case ‘u': 








print("vowel"); 
break; 

case 'y': 

case 'w': print("Sometimes a vowel"); 

break; 
default: print("consonant"); 
} 

} 


} 

} /* Output: 

y. 121: Sometimes a vowel 
n, 110: consonant 

z, 122: consonant 
b 
r 
n 


三 < 


，98: consonant 
，114: consonant 
，116: consonant 


eee OS ee 
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. 121: Sometimes a vowel 
103: consonant 

99: consonant 

102: consonant 

111: vowel 

119: Sometimes a vowel 
122: consonant 


nzonmnom< 


H~ 

由 于 Random.nextInt(26) 会 产生 0 到 26 之 间 的 一 个 值 ， 所 以 在 其 上 加 上 一 个 偏 移 量 “a"， 即 
可 产生 小 写字 母 。 在 case 语 句 中， 使 用 单 引号 引起 的 字符 也 会 产生 用 于 比较 的 整数 值 。 

请 注意 cease 语句 能 够 堆 释 在 一 起 ， 为 一 段 代码 形成 多 重 匹 配 ， 即 只 要 符合 多 种 条 件 中 的 一 
种 ， 就 执行 那 段 特别 的 代码 。 这 时 也 应 注意 将 break 语 句 置 于 特定 case 的 末尾 ， 否 则 控制 流程 会 
简单 地 下 移 ， 处 理 后 面 的 case。 

在 下 面 的 语句 中 : 

int c = rand.nextInt(26) + ‘a’; 
Random.nextInt(0 将 产生 0 一 25 之 间 的 一 个 随机 int 值 ， 它 将 被 加 到 “a” 上 。 这 表示 “a” 将 自 
动 被 转换 为 int 以 执行 加 法 。 为 了 把 c 当 作 字符 打印 ， 必 须 将 其 转型 为 char， 否 则 ， 将 产生 整 型 
输出 。 ; 

#38. (2) 写 一 个 switeh 开 关 语 句 ， 为 每 个 case 打 印 一 个 消息 。 然 后 把 这 个 switch 放 进 for 循 
环 来 测试 每 个 case。 先 让 每 个 case 后 面 都 有 break， 测 试 一 下 会 怎样 ， 然 后 把 break 删 了 ， 看 看 会 
怎样 。 

练习 9: (4) 一 个 斐 波 那 契 数列 是 由 数字 1、1、2、3、5、8、13、21、34 等 等 组 成 的 ， 其 中 
每 一 个 数字 (从 第 三 个 数字 起 ) 都 是 前 两 个 数字 的 和 。 创 建 一 个 方法 ， 接 受 一 个 整数 参数 ， 并 
显示 从 第 一 个 元 素 开始 总 共 由 该 参数 指定 的 个 数 所 构成 的 所 有 斐 波 那 契 数字 。 例 如 ， 如 果 运 行 [53 
java Fibonacci 5 (其 中 Fibonacei 是 类 名 ) ， 那 么 输出 就 应 该 是 1、1、2、3、5。 

练习 10: (5) 吸血 鬼 数 字 是 指 位 数 为 偶数 的 数字 ， 可 以 由 一 对 数字 相 乘 而 得 到 ， 而 这 对 数字 
各 包含 乘积 的 一 半 位 数 的 数字 ， 其 中 从 最 初 的 数字 中 选取 的 数字 可 以 任意 排序 。 以 两 个 0 结尾 的 
数字 是 不 允许 的 ， 例 如 ， 下 列 数字 都 是 “吸血 鬼 ” 数 字 ， 

1260 =21 * 60 

1827 =21 * 87 

2187=27*81 ， 

写 一 个 程序 ， 找 出 4 位 数 的 所 有 吸血 鬼 数字 (Dan Forhan 推 荐 ) 。 


4.9 总 结 


本 章 介 绍 了 大 多 数 编程 语言 都 具有 的 基本 特性 : 运算 、 操 作 符 优先 级 、 类 型 转换 以 及 选择 
和 循环 等 等 。 现 在 ， 我 们 已 经 做 好 了 准备 ， 以 使 自己 更 靠近 面向 对 象 的 程序 设计 的 世界 。 在 下 
一 章 里 ， 我 们 将 讨论 对 象 的 初始 化 与 清理 ， 再 后 面 则 讲述 隐藏 实现 细节 (implementation hiding) 
这 一 核心 概念 。 
所 选 习题 的 答案 都 可 以 在 名 为 The Thinking in Java Annotated Solution Guide 的 电子 文档 中 找 
到 ， 读 者 可 以 从 www.MindView.net 处 购买 此 文档 。 154) 




















第 5 章 初始 化 与 清理 


随 着 计算 机 革命 的 发 展 ,“ 不 安全 ”的 编程 方式 已 逐渐 成 为 编程 代价 高 昂 的 主因 之 一 。 

初始 化 和 清理 (cleanup) 正 是 涉及 安全 的 两 个 问题 。 许 多 C 程 序 的 错误 都 源 于 程序 员 忘 记 
初始 化 变量 。 特 别 是 在 使 用 程序 库 时 ， 如 果 用 户 不 知道 如 何 初始 化 库 的 构件 (或 者 是 用 户 必须 
进行 初始 化 的 其 他 东西 )， 更 是 如 此 。 清 理 也 是 一 个 特殊 问题 ， 当 使 用 完 一 个 元 素 时 ， 它 对 你 也 
就 不 会 有 什么 影响 了 ， 所 以 很 容易 把 它 忘记 。 这 样 一 来 ， 这 个 元 素 占用 的 资源 就 会 一 直 得 不 到 
释放 ， 结 果 是 资源 (尤其 是 内 存 ) 用 尽 。 

C++ 引入 了 构造 器 (constructor) 的 概念 ， 这 是 一 个 在 创建 对 象 时 被 自动 调用 的 特殊 方法 。 
Java 中 也 采用 了 构造 器 ， 并 额外 提供 了 “垃圾 回收 器 "。 对 于 不 再 使 用 的 内 存 资源 ， 垃 圾 回收 器 
能 自动 将 其 释放 。 本 章 将 讨论 初始 化 和 清理 的 相关 问题 ， 以 及 Java 对 它们 提供 的 支持 。 


5.1 用 构造 器 确保 初始 化 


可 以 假想 为 编写 的 每 个 类 都 定义 一 个 initialize0 方 法 。 该 方法 的 名 称 提醒 你 在 使 用 其 对 象 
之 前 ， 应 首先 调用 initialize()。 然 而 ， 这 同时 意味 着 用 户 必 须 记得 自己 去 调用 此 方法 。 在 Java 
中 ， 通 过 提供 构造 器 ， 类 的 设计 者 可 确保 每 个 对 象 都 会 得 到 初始 化 。 创 建 对 象 时 ， 如 果 其 类 
具有 构造 器 ，Java 就 会 在 用 户 有 能 力 操作 对 象 之 前 自动 调用 相应 的 构造 器 ， 从 而 保证 了 初始 化 
的 进行 。 

接 下 来 的 问题 就 是 如 何 命名 这 个 方法 。 有 两 个 问题 : 第 一 ， 所 取 的 任何 名 字 都 可 能 与 类 的 
某 个 成 员 名 称 相 冲 突 ， 第 二 ， 调 用 构造 器 是 编译 器 的 责任 ， 所 以 必须 让 编译 器 知道 应 该 调用 哪 
个 方法 。C++ 语 言 中 采用 的 解决 方案 看 来 最 简单 且 更 符合 逻辑 ， 所 以 在 Java 中 也 采用 了 这 种 方 
R: 即 构造 器 采用 与 类 相同 的 名 称 。 考 虚 到 在 初始 化 期 间 要 自动 调用 构造 器 ， 这 种 做 法 就 顺 理 
成 章 了 。 

以 下 就 是 一 个 带 有 构造 器 的 简单 类 : 


11: initialization/SimpleConstructor.java 
// Demonstration of a simple constructor. 


class Rock { 
Rock() { // This is the constructor 
System.out.print("Rock "); 
} 
} 


public class SimpleConstructor { 
public static void main(String[] args) { 
for(int 1 = @; i < 16; i++) 
new Rock() ; 
} 
} /* Output: 
Rock Rock Rock Rock Rock Rock Rock Rock Rock Rock 
Wh 


现在 ， 在 创建 对 象 时 : 


new Rock() ; 


将 会 为 对 象 分 配 存储 空间 ， 并 调用 相应 的 构造 器 。 这 就 确保 了 在 你 能 操作 对 象 之 前 ， 它 已 经 被 
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恰当 地 初始 化 了 。 

请 注意 ， 由 于 构造 器 的 名 称 必须 与 类 名 完全 相同 ， 所 以 “每 个 方法 首 字母 小 写 ”的 编码 风 
格 并 不 适用 于 构造 器 。 

不 接受 任何 参数 的 构造 器 叫做 默认 构造 器 ，Java 文 档 中 通常 使 用 术语 无 参 构 造 器 ， 但 是 默 
认 构 造 器 在 Java 出 现 之 前 已 经 使 用 许多 年 了 ， 所 以 我 仍旧 倾向 于 使 用 它 。 但 是 和 其 他 方法 一 样 ， 
构造 器 也 能 带 有 形式 参数 ， 以 便 指定 如 何 创建 对 象 。 对 上 述 例子 稍 加 修改 ， 即 可 使 构造 器 接受 
一 个 参数 ， 


/1/: initialfzation/SimpleConstructor2,java 
// Constructors can have arguments. 


class Rock2 { 
Rock2(int i) { 
System.out.print("Rock "+i +" "); 
} 
$ 


public class SimpleConstructor2 { 
public static void main(String[] args) { 
for(int i =; 1 < 8; i++) 
new Rock? (i); 
} 


} /* Output: 
Rock © Rock 1 Rock 2 Rock 3 Rock 4 Rock 5 Rock 6 Rock 7 
Whim 


有 了 构造 器 形式 参数 ， 就 可 以 在 初始 化 对 象 时 提供 实际 参数 。 例 如 ， 假 设 类 Tree 有 一 个 构 
造 器 ， 它 接受 一 个 整 型 变量 来 表示 树 的 高 度 ， 就 可 以 这 样 创建 一 个 Tree 对 象 ， 

Tree t = new Tree(12); // 12-foot tree 

如 果 Treetinb 是 Tree 类 中 唯一 的 构造 器 ， 那 么 编译 器 将 不 会 允许 你 以 其 他 任何 方式 创建 Tree 
对 象 。 

构造 器 有 助 于 减少 错误 ， 并 使 代码 更 易于 阅读 。 从 概念 上 讲 , “初始化 ”与 “创建 ”是 彼此 
独立 的 ， 然 而 在 上 面 的 代码 中 ， 你 却 找 不 到 对 initialize0 方 法 的 明确 调用 。 在 Java 中 ,“ 初 始 化 " 
和 “创建 ”捆绑 在 一 起 ， 两 者 不 能 分 离 。 

构造 器 是 一 种 特殊 类 型 的 方法 ， 因 为 它 没有 返回 值 。 这 与 返回 值 为 空 (void) 明显 不 同 。 
对 于 空 返回 值 ， 尽 管 方法 本 身 不 会 自动 返回 什么 ， 但 仍 可 选择 让 它 返 回 别 的 东西 。 构 造 器 则 不 
会 返回 任何 东西 ， 你 别 无 选择 (new 表 达 式 确实 返回 了 对 新 建 对 象 的 引用 ， 但 构造 器 本 身 并 没有 
任何 返回 值 )。 假 如 构造 器 具有 返回 值 ， 并 且 允 许 人 们 自行 选择 返回 类 型 ， 那 么 势必 得 让 编译 器 
知道 该 如 何 处 理 此 返回 值 。 

练习 1: (1) 创建 一 个 类 ， 它 包含 一 个 未 初始 化 的 String 引 用 。 验 证 该 引用 被 Java 初 始 化 成 
了 null。 

练习 2: (2) 创建 一 个 类 ， 它 包含 一 个 在 定义 时 就 被 初始 化 了 的 String 域 ， 以 及 另 一 个 通过 构 
造 器 初始 化 的 String 域 。 这 两 种 方式 有 何 差异 ? 


5.2 方法 重 载 


任何 程序 设计 语言 都 具备 的 一 项 重要 特性 就 是 对 名 字 的 运用 。 当 创建 一 个 对 象 时 ， 也 就 给 
此 对 象 分 配 到 的 存储 空间 取 了 一 个 名 字 。 所 谓 方 法 则 是 给 某 个 动作 取 的 名 字 。 通 过 使 用 名 字 ， 
你 可 以 引用 所 有 的 对 象 和 方法 。 名 字 起 得 好 可 以 使 系统 更 易于 理解 和 修改 。 就 好 比 写 散文 一 - 目 
的 是 让 读者 易于 理解 。 
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将 人 类 语言 中 存在 细微 差别 的 概念 “映射 ”到 程序 设计 语言 中 时 ， 问 题 随 之 而 生 。 在 日 常 
生活 中 ， 相 同 的 词 可 以 表达 多 种 不 同 的 含义 一 一 它们 被 “ 重 载 ”了 。 特 别 是 含义 之 间 的 差别 很 
小 时 ， 这 种 方式 十 分 有 用 。 你 可 以 说 “清洗 衬衫 "、“ 清 洗车 “清洗 狗 "。 但 如 果 硬 要 这 样 说 就 
显得 很 思春 :“ 以 洗 衬衫 的 方式 洗 衬衫 "、“ 以 洗车 的 方式 洗车 "、“ 以 洗 狗 的 方式 洗 狗 " RRA 
为 听众 根本 不 需要 对 所 执行 的 动作 做 出 明确 的 区 分 。 大 多 数 人 类 语言 具有 很 强 的 “ 宛 余 ”性 ， 
所 以 即使 漏 掉 了 几 个 词 ， 仍 然 可 以 推断 出 含义 。 不 需要 对 年 个 概念 都 使 用 不 同 的 词汇 一 AR 
体 的 语 境 中 就 可 以 推断 出 含义 。 

大 多 数 程序 设计 语言 (尤其 是 C) 要 求 为 每 个 方法 (在 这 些 语言 中 经 常 称 为 函数 ) 都 提供 一 
个 独一无二 的 标识 符 。 所 以 绝 不 能 用 名 为 print0 的 函数 显示 了 整数 之 后 ， 又 用 一 个 名 为 print0 的 
函数 显示 浮 点 数 一 - 每 个 函数 都 要 有 唯一 的 名 称 。 

在 Java (和 C++) 里 ， 构 造 器 是 强制 重 载 方法 名 的 另 一 个 原因 。 了 既然 构造 器 的 名 字 已 经 由 类 
名 所 决定 ， 就 只 能 有 一 个 构造 器 名 。 那 么 要 想 用 多 种 方式 创建 一 个 对 象 该 怎么 办 呢 ? 假设 你 要 
创建 一 个 类 ， 既 可 以 用 标准 方式 进行 初始 化 ， 也 可 以 从 文件 里 读 取信 息 来 初始 化 。 这 就 需要 两 
个 构造 器 : 一 个 默认 构造 器 ， 另 一 个 取 字 符 串 作为 形式 参数 一 该 字符 串 表示 初始 化 对 象 所 需 
的 文件 名 称 。 由 于 都 是 构造 器 ， 所 以 它们 必须 有 相同 的 名 字 ， 即 类 名 。 为 了 让 方法 名 相同 而 形 
式 参数 不 同 的 构造 器 同时 存在 ， 必 须 用 到 方法 重 载 。 同 时 ， 尽 管 方法 重 载 是 构造 器 所 必需 的 ， 
但 它 亦 可 应 用 于 其 他 方法 ， 且 用 法 同样 方便 。 

下 面 这 个 例子 同时 示范 了 重 载 的 构造 器 和 重 载 的 方法 : 


1/: initialization/Overloading. java 
// Demonstration of both constructor 
// and ordinary method overloading. 
import static net.mindview.util.Print.*; 





class Tree { 

int height; 

Tree() { 
print("Planting a seedling"); 
height = 0; 

} 

Tree(int initialHeight) { 
height = initialHeight; 
print("Creating new Tree that is ”+ 

height + ”feet tall"); 


} 
void info() { 
print("Tree is " + height + ”feet tall"); 


} 
void info(String s) { 
print(s + ": Tree is ”+ height + ”feet tall"); 


} 


public class Overloading { 
public static void te ne args) { 


for(int 1 = 0; 1 < 5; i++) 
Tree t = new Tree(i); 
t.info(): 


t.info("overloaded method"); 


} 
// Overloaded constructor: 
new Tree(); 


让 
} /* Output: 
Creating new Tree that is @ feet tall 
Tree is 9 feet tall 
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overloaded method: Tree is @ feet tall 
Creating new Tree that is 1 feet tall 
Tree is 1 feet tall 

overloaded method: Tree is 1 feet tall 
Creating new Tree that is 2 feet tall 
Tree is 2 feet tall 

overloaded method: Tree is 2 feet tall 
Creating new Tree that is 3 feet tall 
Tree is 3 feet tall 

overloaded method: Tree is 3 feet tall 
. Creating new Tree that is 4 feet tall 
Tree is 4 feet tall 

overloaded method: Tree is 4 feet tall 
Planting a seedling 

Whim 


创建 Tree 对 象 的 时 候 ， 既 可 以 不 含 参数 ， 也 可 以 用 树 的 高 度 当 参数 。 前 者 表示 一 棵 树苗 ， 
后 者 表示 已 有 一 定 高 度 的 树木 。 要 支持 这 种 创建 方式 ， 得 有 一 个 默认 构造 器 和 一 个 采用 现 有 高 
度 作为 参数 的 构造 器 。 

或 许 你 还 想 通 过 多 种 方式 调用 info0 方 法 。 例 如 ， 你 想 显示 额外 信息 ， 可 以 用 info(String) 方 
法 ， 没有 的 话 就 用 info0。 要 是 对 明显 相同 的 概念 使 用 了 不 同 的 名 字 ， 那 一 定 会 让 人 很 纳闷。 好 
在 有 了 方法 重 载 ， 可 以 为 两 者 使 用 相同 的 名 字 。 

5.2.1 区 分 重 载 方法 

要 是 几 个 方法 有 相同 的 名 字 ，Java 如 何 才 能 知道 你 指 的 是 哪 一 个 呢 ? 其 实 规则 很 简单 ; 每 
个 重 载 的 方法 都 必须 有 一 个 独一无二 的 参数 类 型 列表 。 

稍 加 思考 ， 就 会 觉得 这 是 合理 的 。 毕 竞 ， 对 于 名 字 相 同 的 方法 ， 除 了 参数 类 型 的 差异 以 外 ， 
还 有 什么 办 法 能 把 它们 区 别 开 呢 ? 

甚至 参数 顺序 的 不 同 也 足以 区 分 两 个 方法 。 不 过 ， 一 般 情 况 下 别 这 么 做 ， 因 为 这 会 使 代码 
难以 维护 : 

//: imitialization/OverloadingOrder.java 

// Overloading based on the order of the arguments. 

import static net.mindview.util.Print.*; 

public class OverloadingOrder { 


static void f(String s, int i) ( 
print("String: "+s +", int: " + i); 


} 
static void f(int 1, String s) { 
print("int: "+ i+", String: * + s); 


} 
public static void main(Stringl] args) { 
f("String first", 11); 
£(99, “Int first"); 
} 
} /* Output: 
String: String first, int: 11 
int: 99, String: Int first 
Wha 


上 例 中 两 个 print0 方 法 虽然 声明 了 相同 的 参数 ， 但 顺序 不 同 ， 因 此 得 以 区 分 。 
5.2.2 涉及 基本 类 型 的 重 载 

基本 类 型 能 从 一 个 “ 较 小 ”的 类 型 自动 提升 至 一 个 “ 较 大 ”的 类 型 ， 此 过 程 一 旦 牵涉 到 重 
载 ， 可 能 会 造成 一 些 混淆 。 以 下 例子 说 明了 将 基本 类 型 传递 给 重 载 方法 时 发 生 的 情况 : 


11: initialization/PrimitiveOverloading. java 
// Promotion of primitives and overloading. 
import static net.mindview.util.Print.*; 
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public 
void 
void 
void 
void 
void 
void 
void 


void 
void 
void 
void 
void 
void 


void 
void 
void 
void 
void 


void 
void 
void 
void 


void 
void 
void 


void 
void 


void 


void 


printnb("5: "); 
#1(5) 5 f2(5) 573 


) 
void 


char x = 'x'; 


class PrimitiveOverloading { 
fl(char x) { printnb("f1(char) "); 
fl(byte x) { printnb("fi(byte) “ 
fl(short x) { printnb("f1(short) "); } 
flint x) { printnb("fl(int) "); } 

fl (long x) { printnb("f1(long) 
fl(float x) { printnb(*f1(float) 
fl(double x) { printnb("f1(double) 

















f2(byte x) { printnb("f2(byte) "); } 
f2(short x) { printnb("f2(short) *); } 
f2Cint x) { printnb("f2(int) "); } 
2(long x) { printnb("f2 (long) 
f2(float x) { printnb("f2(float) 
f2(double x) { printnb("f2(double) “ 









f3(short x) { printnb("f3(short) "); 
f3(int x) { printnb("f3(int) "); } 

f3(long x) { printnb("f3(Long) 
f3(float x) { printnb("f3(float) 
f3(double x) { printnb("f3 (double) 


} 






f4(int x) { printnb("f4(int) "); } 
fa(tong x) { printnb("f4(Long) 
f4(float x) { printnb("f4(float) 
f4(double x) { printnd("f4(double) 









*5(long x) { printnb("f5(long) *) 
5(float x) { printnb("f5(float) 
f5(double x) { printnb("f5(double) * 





f6(float x) { printnd("f6(float) ") 
f6(double x) { printnb("f6 (double) 





f7(double x) { printnb("f7(double) *) 


testConstVal() { 


testChar() { 


printnb("char: "); 


F1(K) 5 #20) ;73(x) 5 £4.00) 5 5 (x) F600) ;17(x); print; 


} 
void 


testByte() { 


byte x = 0; 
printnb("byte: "); 


F1(x) 5 £2 (0) 5 £300) :14(x) 55.0) :6(x) :17(x); print; 


} 
void 


testShort() { 


short x = 0; 
printnb("short: *); 


F100) 5 F200) 5 £300) ;fF4(x) ;15(x);f6(x) 5700; print): 


} 
void 


testint() { 


int x = 6; 
printnb("int: "); 


F(x) 5 #204) 5 £3.00) 54.00) 50x) 5600) F700; printQ: 


} 
void 


testLong() { 


long x = 0; 
printnb("long: "); 


fl1(x);f2(x);f3(x):f4(x):f5(x);f6(x);f7(x); printQ; 


void 


testFloat() { 


float x = 0; 


(5) :74(5) :#5(5) :6(5):f7(5); printO: 
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printnb("float: 
f1(x);f2(x);f3(x):f4(x):f5(x):f6(x);f7(x); print: 


} 

void testDouble() { 
double x = 0; 
printnb("double: "); 
fl1(x);f2(x);f3(x);f4(x):f5(x):f6(x);f7(x); printO: 





public static void main(String[] args) { 
PrimitiveOverloading p = 

new PrimitiveOverloading(): 

testConstVal(): 

testChar(); 

testByte(); 

testShort(); 

testInt(); 

testLong(); 

testFloat(); 

testDouble(); 


aooooa 


) /* Output: 

5: f1(int) f2(int) f3(int) f4(int) f5(long) f6(float) 
f7(double) 

char: f(char) f2(int) f3(int) f4(int) f5(long) f6(float) 
f7(double) 

byte: fl(byte) f2(byte) f3(short) f4(int) f5(long) 
f6(float) f7 (double) 

short: fl(short) f2(short) f3(short) f4(int) f5(long) 
6(float) {7(double) 

int: f1(int) f2(int) f3(int) f4(int) f5(long) f6(float) 
7(double) 

long: f1(long) f2(long) f3(long) f4(long) f5(long) 
f6(float) f7(double) 

float: fl(float) f2(float) f3(float) f4(float) f5(float) 
f6(float) f7 (double) 

double: fi(double) f2(double) f3(double) f4(double) 
f5(double) f6(double) f7(double) ` 

Whim 


你 会 发 现 常数 值 5 被 当 作 int 值 处 理 ， 所 以 如 果 有 某 个 重 载 方法 接受 int 型 参数 ， 它 就 会 被 调 
用 。 至 于 其 他 情况 ， 如 果 传 人 的 数据 类 型 (实际 参数 类 型 ) 小 于 方法 中 声明 的 形式 参数 类 型 ， 
实际 数据 类 型 就 会 被 提升 。ehar 型 略 有 不 同 ， 如 果 无 法 找到 恰好 接受 char 参 数 的 方法 ， 就 会 把 
char 直 接 提升 至 int 型 。 

如 果 传人 的 实际 参数 大 于 重 载 方法 声明 的 形式 参数 ， 会 出 现 什么 情况 呢 ? 修改 上 述 程序 ， 
就 能 得 到 答案 : 


/1: initialization/Demotion. java 
// Demotion of primitives and overloading. 
import static net.mindview.util.Print.*; 





public class Demotion { 
void fl(char x) { print("f1(char)"); } 
void fl(byte x) { print("fl(byte)") 
void fl(short x) { print("f1(short)") 
void fl(int x) { print("fiCint)"); } 
void fl(long x) { print("f1(long)"); } 
void fl(float x) { print("f1(float)"); 
void fl(double x) { print("f1(double) 











void f2(char x) { print("f2(char)"); } 
void f2(byte x) { print("f2(byte)"); } 
void f2(short x) { print("f2ishort)") 
void f2(int x) { print("f2(int)"); } 
void f2(long x) { print("f2(long)"); } 


} 
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void f2(float x) { print("f2(float)"); } 


void f3(char x) { print("f3(char) 
void f3(byte x) { print("f3 (byte) 
void f3(short x) { print("f3(short)" 
void f3(int x) { print("f3(int)"); } 
void f3(long x) { print("f3(long)"): } 





void f4(char x) { print("f4(char)"): } 
void f4(byte x) ( print("f4(byte) 
void f4(short x) { print("f4(shor 
void f4(int x) { print("f4(int)") 
void fS(char x) { print("f5(char) 
void f5(byte x) { print("f5(byte)"); 
void f5(short x) { print("fS(short)"); } 





void f6(char x) { print("f6(char)"); } 
void f6(byte x) { print("f6(byte)"); } 


void f7(char x) { print("f7(char)"); } 


void testDouble() { 
double x = 0; 
print("double argument: "); 
F1 (x); f2((float)x) :#3((Long)x) : f4((int)x) ; 
#5((short)x) ; £6((byte)x) :f7((char)x) ; 

} 

Public static void main(String[] args) { 
Demotion p = new Demotion(); 
p.testDouble(); 


} 
} /* Output: 
double argument: 
f1(double) 
f2(float) 
£3(1ong) 
talint) 
f5(short) 
f6(byte) 
f7(char) 
Whim 


在 这 里 ， 方 法 接受 较 小 的 基本 类 型 作为 参数 。 如 果 传 入 的 实际 参数 较 大 ， 就 得 通过 类 型 转 
换 来 执行 窗 化 转换 。 如 果 不 这 样 做 ， 编 译 器 就 会 报错 。 
5.2.3 以 返回 值 区 分 重 载 方法 

读者 可 能 会 想 :“ 在 区 分 重 载 方法 的 时 候 ， 为 什么 只 能 以 类 名 和 方法 的 形 参 列表 作为 标准 
We? 能 否 考虑 用 方法 的 返回 值 来 区 分 呢 ? ”比如 下 面 两 个 方法 ， 虽 然 它们 有 同样 的 名 字 和 形式 
参数 ， 但 却 很 容易 区 分 它们 : 


void fO D 
int fO { return 1; } 


只 要 编译 器 可 以 根据 语 境 明确 判断 出 语义 ， 比 如 在 int x=f0 中 ， 那 么 的 确 可 以 据 此 区 分 重 
载 方法 。 不 过 ， 有 时 你 并 不 关心 方法 的 返回 值 ， 你 想 要 的 是 方法 调用 的 其 他 效果 (这 常 被 称 
为 “为 了 副作用 而 调用 ”)， 这 时 你 可 能 会 调用 方法 而 忽略 其 返回 值 。 所 以 ， 如 果 像 下 面 这 样 
调用 方法 : 

t0: 

此 时 Java 如 何 才能 判断 该 调用 哪 一 个 f0 呢 ?别人 该 如 何 理解 这 种 代码 呢 ? 因此 ， 根 据 方法 
的 返回 值 来 区 分 重 载 方法 是 行 不 通 的 。 
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5.3 默认 构造 器 


如 前 所 述 ， 默 认 构造 器 (又 名 “无 参 ” 构 造 器 ) 是 没有 形式 参数 的 一 一 它 的 作用 是 创建 一 个 
“默认 对 象 "。 如 果 你 写 的 类 中 没有 构造 器 ， 则 编译 器 会 自动 帮 你 创建 一 个 默认 构造 器 。 例 如 : 
1/: initialization/DefaultConstructor. java 


class Bird {} 


public class DefaultConstructor { 
Public static void main(String{] args) { 
Bird b = new Bird(); // Default! 


} Mine 

PEK 

new Bird() 

行 创建 了 一 个 新 对 象 ， 并 调用 其 默认 构造 器 一 一 即使 你 没有 明确 定义 它 。 没 有 它 的 话 ， 就 
没有 方法 可 调用 ， 就 无 法 创建 对 象 。 但 是 ;如 果 已 经 定义 了 一 个 构造 器 (无 论 是 否 有 参数 )， 编 
译 器 就 不 会 帮 你 自动 创建 默认 构造 器 : 


1/: Anitialization/NoSynthesis. java 


class Bird2 { 
Bird2(int 1) {} 
Bird2(double d) {} 


3 


public class NoSynthesis { 
public static void main(String[] args) { 
//! Bird? b = new Bird2(); // No default 
Bird2 b2 = new Bird2(1); 
Bird2 b3 = new Bird2(1.0); 


) hh 

要 是 你 这 样 写 : 

new Bird2() 
编译 器 就 会 报错 : 没有 找到 匹配 的 构造 器 。 这 就 好 比 ， 要 是 你 没有 提供 任何 构造 器 ， 编 译 
器 会 认为 “你 需要 一 个 构造 器 ， 让 我 给 你 制造 一 个 吧 ”， 但 假如 你 已 写 了 一 个 构造 器 ， 编 
译 器 则 会 认为 “ 啊 ， 你 已 写 了 一 个 构造 器 ， 所 以 你 知道 你 在 做 什么 ， 你 是 刻意 省 略 了 默认 
构造 器 。” 

练习 3，(1) 创建 一 个 带 默 认 构造 器 ( 即 无 参 构造 器 ) 的 类 ， 在 构造 器 中 打印 一 条 消息 。 为 
这 个 类 创建 一 个 对 象 。 

练习 4: (1) 为 前 一 个 练习 中 的 类 添加 一 个 重 载 构造 器 ， 令 其 接受 一 个 字符 串 参 数 ， 并 在 构 
造 器 中 把 你 自己 的 消息 和 接收 的 参数 一 起 打印 出 来 。 

练习 5: (2) 创建 一 个 名 为 Dog 的 类 ， 它 具有 重 载 的 bark0 方 法 。 此 方法 应 根据 不 同 的 基本 数 
据 类 型 进行 重 载 ， 并 根据 被 调用 的 版 本 ， 打 印 出 不 同类 型 的 狗 喘 (barking), Mm (howling) 
等 信息 。 编 写 main0 来 调用 所 有 不 同 版 本 的 方法 。 

练习 6: (1) 修改 前 一 个 练习 的 程序 ， 让 两 个 重 载 方法 各 自 接受 两 个 类 型 的 不 同 的 参数 ， 但 
二 者 顺序 相反 。 验 证 其 是 否 工作 。 

练习 7: (1) 创建 一 个 没有 构造 器 的 类 ， 并 在 main0 中 创建 其 对 象 ， 用 以 验证 编译 器 是 否 真 
的 自动 加 入 了 默认 构造 器 。 





84 HSK 





5.4 this 关 键 字 
如 果 有 同一 类 型 的 两 个 对 象 ， 分 别 是 a 和 b。 你 可 能 想 知道 ， 如 何 才能 让 这 两 个 对 象 都 能 调 


用 peel0 方 法 呢 ， 





/1: initialization/BananaPeel. java 
class Banana { void peel(int i) { /* ... */ } } 


public class BananaPeel { 
public static void main(String[] args) { 
Banana a = new Banana(), 
b = new Banana(); 
a-peel(1); 
b.peel (2); 


} 
y Mi~ 


如 果 只 有 一 个 peel0 方 法 ， 它 如 何 知道 是 被 3 还 是 被 b 所 调用 的 呢 ? 

为 了 能 用 简便 、 面 向 对 象 的 语法 来 编写 代码 一 - 即 “ 发 送 消息 给 对 象 "， 编 译 器 做 了 一 些 藉 
后 工作 。 它 暗自 把 “所 操作 对 象 的 引用 ”作为 第 一 个 参数 传递 给 peel0。 所 以 上 述 两 个 方法 的 调 
用 就 变 成 了 这 样 ， 


Banana.peel(a, 1); 
Banana.peel(b, 2); 


这 是 内 部 的 表示 形式 。 我 们 并 不 能 这 样 书写 代码 ， 并 试图 通过 编译 ， 但 这 种 写法 的 确 能 帮 
你 了 解 实际 所 发 生 的 事情 。 

假设 你 希望 在 方法 的 内 部 获得 对 当前 对 象 的 引用 。 由 于 这 个 引用 是 由 编译 器 “偷偷 ”传人 
的 ， 所 以 没有 标识 符 可 用 。 但 是 ， 为 此 有 个 专门 的 关键 字 ，this。this 关 键 字 只 能 在 方法 内 部 使 
用 ， 表 示 对 “调用 方法 的 那个 对 象 ” 的 引用 。this 的 用 法 和 其 他 对 象 引用 并 无 不 同 。 但 要 注意 ， 
如 果 在 方法 内 部 调用 同一 个 类 的 另 一 个 方法 ， 就 不 必 使 用 this， 直 接 调用 即 可 。 当 前 方法 中 的 
this 引 用 会 自动 应 用 于 同一 类 中 的 其 他 方法 。 所 以 可 以 这 样 写 代码 ; 


/1/: initialization/Apricot. java 
public class Apricot { 


void pick() { /* ... */} 
void pit() { pick(); /* ... */ } 
dM hm 


在 pit0 内 部 ， 你 可 以 写 this.pick0， 但 无 此 必要 。 。 编 译 器 能 帮 你 自动 添加 。 只 有 当 需 要 明 
确 指出 对 当前 对 象 的 引用 时 ， 才 需要 使 用 this 关 键 字 。 例 如 ， 当 需要 返回 对 当前 对 象 的 引用 时 ， 
就 常常 在 return 语 句 里 这 样 写 : 


/1: initialization/Leaf.java 
// Simple use of the “this” keyword. 


public class Leaf { 
int 1 = 0; 
Leaf increment() { 
itt; 
feturn this; 
} 


O 有 些 人 执意 将 this 放 在 每 一 个 方法 调用 和 字段 引用 前 ， 认 为 这 样 “ 更 清楚 更 明确 "。 但 是 ， 千 万 别 这 么 做 。 我 
们 使 用 高 级 语言 的 原因 之 一 就 是 它们 能 帮 我 们 做 一 些 事情 。 要 是 你 把 this 放 在 一 些 没 必要 的 地 方 ， 就 会 使 读 你 
程序 的 人 不 知 所 措 ， 六 为 别人 写 的 代码 不 会 到 处 使 用 this。 人 们 期 望 只 在 必要 处 使 用 this。 遵 循 一 种 一 致 而 直 
观 的 编程 风格 能 节省 时 间 和 人 金钱 。 
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void print() { 
System.out.printin("i = " + i); 


} 
public static void main(String[] args) { 
Leaf x = new Leaf(); 
x. increment () . increment () . increment () .print(); 


} 
} /* Output: 
i=3 
Wh 


由 于 inerement0 通 过 this 关 键 字 返回 了 对 当前 对 象 的 引用 ， 所 以 很 容易 在 一 条 语句 里 对 同一 
个 对 象 执行 多 次 操作 。 

this 关 键 字 对 于 将 当前 对 象 传递 给 其 他 方法 也 很 有 用 : 

//: initiatization/PassingThis.java 


class Person { 


public void eat(Apple apple) { 
Apple peeled = apple. getPeeled(); 
System.out.printin("Yummy") ; 
} 
} 


class Peeler { 
static Apple peel(Apple apple) { 
// ... remove peel 
return apple; // Peeled 


} 


class Apple { 
Apple getPeeled() { return Peeler.peel(this); } 
} 


public class PassingThis { 
public static void main(String(] args) { 
new Person().eat(new Apple()); 


$ 
} /* Output: 1 
Yummy 
Wh 


Apple 需 要 调用 Peelerpeei0 方 法 ， 它 是 一 个 外 部 的 工具 方法 ， 将 执行 由 于 某 种 原因 而 必须 放 
在 Apple 外 部 的 操作 (也许 是 因为 该 外 部 方法 要 应 用 于 许多 不 同 的 类 , 而 你 却 不 想 重复 这 些 代码 ) 。 
为 了 将 其 自身 传递 给 外 部 方法 ，Apple 必 须 使 用 this 关 键 字 。 

练习 8: (1) 编写 具有 两 个 方法 的 类 ， 在 第 一 个 方法 内 调用 第 二 个 方法 两 次 :第 一 次 调用 时 
不 使 用 this 关 键 字 ， 第 二 次 调用 时 使 用 this 关 键 字 一 一 这 里 只 是 为 了 验证 它 是 起 作用 的 ， 你 不 应 
该 在 实践 中 使 用 这 种 方式 。 
5.4.1 在 构造 器 中 调用 构造 器 

可 能 为 一 个 类 写 了 多 个 构造 器 ， 有 时 可 能 想 在 一 个 构造 器 中 调用 另 一 个 构造 器 ， 以 避免 重 
复 代码 。 可 用 this 关 键 字 做 到 这 一 点 。 

通常 写 this 的 时 候 ， 都 是 指 “ 这 个 对 象 ” 或 者 “当前 对 象 "， 而 且 它 本 身 表示 对 当前 对 象 的 
引用 。 在 构造 器 中 ， 如 果 为 this 添 加 了 参数 列表 ， 那 么 就 有 了 不 同 的 含义 。 这 将 产生 对 符合 此 参 
数列 表 的 某 个 构造 器 的 明确 调用 ， 这 样 ， 调 用 其 他 构造 器 就 有 了 直接 的 途径 : 


//: initialization/Flower.java 
// Calling constructors with "this" 
import static net.mindview.util.Print.*; 
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public class Flower { 
int petalCount = 6; 
String s = “initial value": 
Flower(int petals) { 
petalCount = petals; 
print("Constructor w/ int arg only, petalCount= " 
+ petalCount); 


} 

Flower (String ss) { 
print("Constructor w/ String arg only, s = * + ss); 
s = ss; 


} 
Flower(String s, int petals) { 
this(petals); 
ay this(s); // Can't call two! 
this.s = s; // Another use of “this” 
print("String & int args"); 


Flower() { 
this("hi", 47): 
print("default constructor (no args)"); 


t 
void printPetalCount() { 
//! this(11); // Not inside non-constructor! 
print("petalCount = " + petalCount + * s = "+ s); 


pubtic static void main(String[] args) { 
Flower x = new Flower(); 
x.printPetalCount() ; 

} . Output: 

Constructor w/ int arg only, petalCount= 47 

String & int args 

default constructor (no args) 

petalCount = 47 s = hi 

Wh 

构造 器 Flower(String s,int petals) 22]. 尽管 可 以 用 this 调 用 一 个 构造 器 , 但 却 不 能 调用 两 个 。 
此 外 ， 必 须 将 构造 器 调用 置 于 最 起 始 处 ， 否 则 编译 器 会 报错 。 

这 个 例子 也 展示 了 this 的 另 一 种 用 法 。 由 于 参数 s 的 名 称 和 数据 成 员 s 的 名 字 相同 ， 所 以 会 产 
生 歧 义 。 使 用 this.s 来 代表 数据 成 员 就 能 解决 这 个 问题 。 在 Java 程 序 代码 中 经 常 出 现 这 种 写法 ， 
本 书 中 也 常 这 么 写 。 

PprintPetalCount0 方 法 表明 ， 除 构造 器 之 外 ， 编 译 器 禁止 在 其 他 任何 方法 中 调用 构造 器 。 

练习 9: (1) 编 写 具有 两 个 〈 重 载 ) 构造 器 的 类 ， 并 在 第 一 个 构造 器 中 通过 this 调 用 第 二 个 构 
造 器 。 
5.4.2 static 的 含义 

了 解 this 关 健 字 之 后 ， 就 能 更 全 面 地 理解 static (静态 ) 方法 的 含义 。static 方 法 就 是 没有 this 
的 方法 。 在 static 方 法 的 内 部 不 能 调用 非 静态 方法 ， 反 过 来 倒是 可 以 的 。 而 且 可 以 在 没有 创建 
任何 对 象 的 前 提 下 ， 仅 仅 通过 类 本 身 来 调用 static 方 法 。 这 实际 上 正 是 static 方 法 的 主要 用 途 。 它 
很 像 全 局 方法 。Java 中 禁止 使 用 全 局 方法 ， 但 你 在 类 中 置 和 static 方 法 就 可 以 访问 其 他 static 方 法 
和 static 域 。 


有 些 人 认为 static 方 法 不 是 “面向 对 象 ”的 ， 因 为 它们 的 确 具 有 全 局 函数 的 语义 ， 使 用 static 


O 这 不 是 完全 不 可 能 。 如 果 你 传递 一 个 对 象 的 引用 到 静态 方法 里 (静态 方法 可 以 创建 其 自身 的 对 象 )， 然 后 通过 


这 个 引用 (和 this 效 果 相同 ) ， 你 就 可 以 调用 非 静 态 方法 和 访问 非 静 态 数据 成 员 了 。 但 通常 要 达到 这 样 的 效果 ， 
你 只 需 写 一 个 非 静 态 方法 即 可 。 
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方法 时 ， 由 于 不 存在 this， 所 以 不 是 通过 “向 对 象 发 送 消息 ”的 方式 来 完成 的 。 的 确 ， 要 是 在 代 
码 中 出 现 了 大 量 的 static 方 法 ， 就 该 重新 考虑 自己 的 设计 了 。 然 而 ，static 的 概念 有 其 实用 之 处 ， 
许多 时 候 都 要 用 到 它 。 至 于 它 是 否 真 的 “面向 对 象 ”， 就 留 给 理论 家 去 讨论 吧 。 事 实 上 ， 
Smalltalk 语 言 里 的 “类 方法 ”就 是 与 static 方 法 相对 应 的 。 


5.5 清理 : 终结 处 理 和 垃圾 回收 


程序 员 都 了 解 初 始 化 的 重要 性 ， 但 常常 会 忘记 同样 也 重要 的 清理 工作 。 毕 竟 ， 谁 需要 清理 
一 个 int 呢 ? 但 在 使 用 程序 库 时 ， 把 一 个 对 象 用 完 后 就 “ 弃 之 不 顾 ” 的 做 法 并 非 总 是 安全 的 。 当 
然 ，Java 有 垃圾 回收 器 负责 回收 无 用 对 象 占据 的 内 存 资源 。 但 也 有 特殊 情况 : 假定 你 的 对 象 
(并 非 使 用 new) 获得 了 一 块 “特殊 ”的 内 存 区 域 ， 由 于 垃圾 回收 器 只 知道 释放 那些 经 由 new 分 
配 的 内 存 ， 所 以 它 不 知道 该 如 何 释放 该 对 象 的 这 块 “ 特 殊 ”内 存 。 为 了 应 对 这 种 情况 ，Java 允 
许 在 类 中 定义 一 个 名 为 finalize0) 的 方法 。 它 的 工作 原理 “假定 ”是 这 样 的 : 一旦 垃圾 回收 器 准 
备 好 释放 对 象 占用 的 存储 空间 ， 将 首先 调用 其 finalize0 方 法 ， 并 且 在 下 一 次 垃圾 回收 动作 发 生 
时 ， 才 会 真正 回收 对 象 占用 的 内 存 。 所 以 要 是 你 打算 用 finalize0 ， 就 能 在 垃圾 回收 时 刻 做 一 些 
重要 的 清理 工作 。 

这 里 有 一 个 潜在 的 编程 陷阱 ， 因 为 有 些 程序 员 (特别 是 C++ 程序 员 ) 刚 开始 可 能 会 误 把 
finalizeO 当 作 C++ 中 的 析 构 函数 〈C++ 中 销毁 对 象 必 须 用 到 这 个 函数 ) 。 所 以 有 必要 明确 区 分 一 
F; 在 C++ 中 ， 对 象 一 定 会 被 销毁 (如 果 程序 中 没有 缺陷 的 话 ) ， 而 Java 里 的 对 象 却 并 非 总 是 被 
垃圾 回收 。 或 者 换 句 话说 : 

1. 对 象 可 能 不 被 垃圾 回收 。 

2. 垃圾 回收 并 不 等 于 “ 析 构 ”。 

牢记 这 些 ， 就 能 远离 困扰 。 这 意味 着 在 你 不 再 需要 某 个 对 象 之 前 ， 如 果 必 须 执行 某 些 动作 ， 
那么 你 得 自己 去 做 。Java 并 未 提供 “ 析 构 函数 ”或 相似 的 概念 ， 要 做 类 似 的 清理 工作 ， 必 须 自 
已 动 手 创建 一 个 执行 清理 工作 的 普通 方法 。 例 如 ， 假 设 某 个 对 象 在 创建 过 程 中 会 将 自己 绘制 到 
屏幕 上 ， 如 果 不 是 明确 地 从 屏幕 上 将 其 擦 除 ， 它 可 能 永远 得 不 到 清理 。 如 果 在 finalize0 里 加 入 
某 种 擦 除 功能 ， 当 “垃圾 回收 ”发 生 时 (不 能 保证 一 定 会 发 生 ) ，finalize0 得 到 了 调用 ， 图 像 就 
会 被 擦 除 。 要 是 “垃圾 回收 ”没有 发 生 ， 图 像 就 会 一 直 保 留 下 来 。 

也 许 你 会 发 现 ， 只 要 程序 没有 濒临 存储 空间 用 完 的 那 一 刻 ， 对 象 占用 的 空间 就 总 也 得 不 到 
释放 。 如 果 程序 执行 结束 ， 并 且 垃圾 回收 器 一 直 都 没有 释放 你 创建 的 任何 对 象 的 存储 空间 ， 则 
随 着 程序 的 退出 ， 那 些 资源 也 会 全 部 交还 给 操作 系统 。 这 个 策略 是 恰当 的 ， 因 为 垃圾 回收 本 身 
也 有 开销 ， 要 是 不 使 用 它 ， 那 就 不 用 支付 这 部 分 开销 了 。 

5.5.1 finalize() 的 用 途 何在 

此 时 ， 读 者 已 经 明白 了 不 该 将 finalize0 作 为 通用 的 清理 方法 。 那 么 ，finalize0 的 真正 用 途 是 
什么 呢 ? 

这 引出 了 要 记 住 的 第 三 点 : 

3. 垃圾 回收 只 与 内 存 有 关 。 

也 就 是 说 ， 使 用 垃圾 回收 器 的 唯一 原因 是 为 了 回收 程序 不 再 使 用 的 内 存 。 所 以 对 于 与 垃圾 
回收 有 关 的 任何 行为 来 说 (尤其 是 finalize0 方 法 )， 它 们 也 必须 同 内 存 及 其 回收 有 关 。 

但 这 是 否 意味 着 要 是 对 象 中 含有 其 他 对 象 ，finalize0 就 应 该 明确 释放 那些 对 象 呢 ? 不 ,无 
论 对 象 是 如 何 创建 的 ， 垃 圾 回收 器 都 会 负责 释放 对 象 占据 的 所 有 内 存 。 这 就 将 对 finalize0 的 需 
求 限制 到 一 种 特殊 情况 ， 即 通过 某 种 创建 对 象 方式 以 外 的 方式 为 对 象 分 配 了 存储 空间 。 不 过 ， 








172, 











173| 

















174) 








88 RSE 





读者 也 看 到 了 ，Java 中 一 切 皆 为 对 象 ， 那 这 种 特殊 情况 是 怎么 回 事 呢 ? 

看 来 之 所 以 要 有 finalize0， 是 由 于 在 分 配 内 存 时 可 能 采用 了 类 似 C 语 言 中 的 做 法 ， 而 非 Java 
中 的 通常 做 法 。 这 种 情况 主要 发 生 在 使 用 “本 地 方法 ”的 情况 下 ,本 地 方法 是 一 种 在 Java 中 调 
用 非 Java 代 码 的 方式 (关于 本 地 方法 的 讨论 见 本 书 电子 版 第 2 版， 在 www.MindView.net 网 站 上 有 
收录 )。 本 地 方法 目前 只 支持 C 和 C++， 但 它们 可 以 调用 其 他 语言 写 的 代码 ， 所 以 实际 上 可 以 调 
用 任何 代码 。 在 非 Java 代 码 中 ， 也 许 会 调用 C 的 mallocO 函 数 系列 来 分 配 存储 空间 ， 而 且 除非 调 
用 了 free0 函 数 ， 否 则 存储 空间 将 得 不 到 释放 ， 从 而 造成 内 存 泄露 。 当 然 ，free0 是 C 和 C++ 中 的 
函数 ， 所 以 需要 在 finalize0 中 用 本 地 方法 调用 它 。 

至 此 ， 读 者 或 许 已 经 明白 了 不 要 过 多 地 使 用 finalize0 的 道理 了 * 。 对 ， 它 确实 不 是 进行 普通 
的 清理 工作 的 合适 场所 。 那 么 ， 普 通 的 清理 工作 应 该 在 哪里 执行 呢 ? 
5.5.2 你 必须 实施 清理 

要 清理 一 个 对 象 ， 用 户 必须 在 需要 清理 的 时 刻 调用 执行 清理 动作 的 方法 。 这 听 起 来 似乎 很 
简单 ,但 却 与 C++ 中 的 “ 析 构 函数 ”的 概念 稍 有 抵触 。 在 C++ 中 ， 所 有 对 象 都 会 被 销毁 ,或 者 说 ， 
应 该 被 销毁 。 如 果 在 C++ 中 创建 了 一 个 局 部 对 象 (也 就 是 在 堆栈 上 创建 ， 这 在 Java 中 行 不 通 )， 
此 时 的 销毁 动作 发 生 在 以 “ 右 花 括号 ”为 边界 的 、 此 对 象 作 用 域 的 末尾 处 。 如 果 对 象 是 用 new 创 
建 的 (类 似 于 Java 中 )， 那 么 当 程序 员 调 用 C++ 的 delete 操 作 符 时 (Java 没 有 这 个 命令 )， 就 会 调用 
相应 的 析 构 函数 。 如 果 程 序 员 忘 记 调用 delete， 那 么 永远 不 会 调用 析 构 函数 ， 这 样 就 会 出 现 内 存 
泄露 ， 对 象 的 其 他 部 分 也 不 会 得 到 清理 。 这 种 缺陷 很 难 跟 踪 ， 这 也 是 让 C++ 程序 员 转 向 Java 的 一 
个 主要 因素 。 

相反 ，Java 不 允许 创建 局 部 对 象 ， 必 须 使 用 new 创 建 对 象 。 在 Java 中 ， 也 没有 用 于 释放 对 象 
的 delete， 因 为 垃圾 回收 器 会 帮助 你 释放 存储 空间 。 甚 至 可 以 肤浅 地 认为 ， 正 是 由 于 垃圾 收集 机 
制 的 存在 ， 使 得 Java 没 有 析 构 函数 。 然 而 ， 随 着 学 习 的 深入 ， 读 者 就 会 明白 垃圾 回收 器 的 存在 
并 不 能 完全 代替 析 构 函数 。( 而 且 绝对 不 能 直接 调用 finalizeO， 所 以 这 也 不 是 一 种 解决 方案 。) 
如 果 和 希望 进行 除 释放 存储 空间 之 外 的 清理 工作 ， 还 是 得 明确 调用 某 个 恰当 的 Java 方 法 。 这 就 等 
同 于 使 用 析 构 函数 了 ， 只 是 没有 它 方便 。 

记 住 ， 无 论 是 “垃圾 回收 ”还 是 “终结 "， 都 不 保证 一 定 会 发 生 。 如 果 Java 虚 拟 机 (IVM) 
并 未 面临 内 存 耗 尽 的 情形 ， 它 是 不 会 浪费 时 间 去 执行 垃圾 回收 以 恢复 内 存 的 。 
5.5.3 终结 条 件 

通常 ， 不 能 指望 finalize0 ， 必 须 创建 其 他 的 “清理 ”方法 ， 并 且 明确 地 调用 它们 。 看 来 ， 
finalize0 只 能 存在 于 程序 员 很 难 用 到 的 一 些 降 涩 用 法 里 了 。 不过， finalize0 还 有 一 个 有 趣 的 用 法 ， 
它 并 不 依赖 于 每 次 都 要 对 finalize0O 进 行 调用 ， 这 就 是 对 象 终结 条 件 的 验证 。 

当 对 某 个 对 象 不 再 感 兴趣 一 一 也 就 是 它 可 以 被 清理 了 ， 这 个 对 象 应 该 处 于 某 种 状态 ， 使 它 占 
用 的 内 存 可 以 被 安全 地 释放 。 例 如 ， 要 是 对 象 代表 了 一 个 打开 的 文件 ， 在 对 象 被 回收 前 程序 员 
应 该 关闭 这 个 文件 。 只 要 对 象 中 存在 没有 被 适当 清理 的 部 分 ， 程序 就 存在 很 隐 星 的 缺陷 。 
finalizeO 可 以 用 来 最 终 发 现 这 种 情况 一 一 尽管 它 并 不 总 是 会 被 调用 。 如 果 某 次 finalize0 的 动作 使 
得 缺陷 被 发 现 ， 那 么 就 可 据 此 找 出 问题 所 在 一 一 这 才 是 人 们 真正 关心 的 。 

以 下 是 个 简单 的 例子 ， 示 范 了 finalize0 可 能 的 使 用 方式 : 


© Joshua Bloch 在 题 为 “避免 使 用 终结 函数 ”一 节 中 走 的 更 远 ， 他 提 到 :“ 终 结 函数 无 法 预料 ， 常 常 是 危险 的 ， 
总 之 是 多 余 的 ,”《Effective Java》, 第 20 页 ，(Addison-Wesley 2001), 
© 这 个 术语 是 在 由 Bill Venners (wwwartima.com) 和 我 一 同 开 的 培训 班 上 发 明 的 。 
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//: initialization/TerminationCondition.java 
// Using finalize() to detect an object that 
// hasn't been properly cleaned up. 


class Book { 
boolean checkedOut = false; 
Book(boolean checkOut) { 
checkedOut = checkOut; 


} 
void checkIn() { 
checkedOut = false; 
} 
protected void finalize() { 
if (checkedOut) 
System.out.printin("Error: checked out"); 
// Normally, you'll also do this: 
// super.finalize(); // Call the base-class version 
} 
} 


public class TerminationCondition { 
public static void main(String{] args) { 

Book novel = new Book(true) ; 
// Proper cleanup: 
novel .checkIn() ; 
// Drop the reference, forget to clean up: 
new Book(true) ; 
// Forée garbage collection & finalization: 
System.gc(); 


} 
} /* Output: 


Error: checked out 
Whe 


本 例 的 终结 条 件 是 : 所 有 的 Book 对 象 在 被 当 作 垃圾 回收 前 都 应 该 被 签 入 (check in)。 但 在 
main0 方 法 中 ， 由 于 程序 员 的 错误 ， 有 一 本 书 未 被 签 入 。 要 是 没有 finalize0 来 验证 终结 条 件 ， 将 
很 难 发 现 这 种 缺陷 。 

注意 ，System.ge0 用 于 强制 进行 终结 动作 。 即 使 不 这 么 做 ， 通 过 重复 地 执行 程序 (假设 程 
序 将 分 配 大 量 的 存储 空间 而 导致 垃圾 回收 动作 的 执行 )， 最 终 也 能 找 出 错误 的 Book 对 象 。 

你 应 该 总 是 假设 基 类 版 本 的 finalize0 也 要 做 某 些 重要 的 事情 ， 因 此 要 使 用 super 来 调用 它 ， 
就 像 在 Book.finalize0 中 看 到 的 那样 。 在 本 例 中 ， 它 被 注释 掉 了 ， 因 为 它 需 要 进行 异常 处 理 ， 而 
我 们 还 没有 介绍 过 这 部 分 内 容 。 

练习 10: (2) 编写 具有 finalize0 方 法 的 类 ， 并 在 方法 中 打印 消息 。 在 main0 中 为 该 类 创建 一 
个 对 象 。 试 解释 这 个 程序 的 行为 。 

练习 11:(4) 修改 前 一 个 练习 的 程序 ， 让 你 的 finalize0 总 会 被 调用 。 

练习 12: (4) 编写 名 为 Tank 的 类 ， 此 类 的 状态 可 以 是 “ 满 的 ”或 “ 空 的 "。 其 终结 条 件 是 ; 
对 象 被 清理 时 必须 处 于 空 状态 。 请 编写 finalizeO 以 检验 终结 条 件 是 否 成 立 。 在 main0 中 测试 
Tank 可 能 发 生 的 几 种 使 用 方式 。 

5.5.4 垃圾 回收 器 如 何 工作 

在 以 前 所 用 过 的 程序 语言 中 ， 在 堆 上 分 配对 象 的 代价 十 分 高 昂 ， 因 此 读者 自然 会 觉得 Java 
中 所 有 对 象 (基本 类 型 除外 ) 都 在 堆 上 分 配 的 方式 也 非常 高 兄 。 然 而 ， 垃 圾 回收 器 对 于 提高 对 
象 的 创建 速度 ， 却 具有 明显 的 效果 。 听 起 来 很 奇怪 一 -存储 空间 的 释放 竟然 会 影响 存储 空间 的 
分 配 ， 但 这 确实 是 某 些 Java 虚 拆 机 的 工作 方式 。 这 也 意味 着 ，Java 从 堆 分 配 空间 的 速度 ， 可 以 和 
其 他 语言 从 堆栈 上 分 配 空间 的 速度 相 巡 美 。 
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打 个 比方 ， 你 可 以 把 C++ 里 的 堆 想像 成 一 个 院子 ， 里 面 每 个 对 象 都 负责 管理 自己 的 地 盘 。 
一 段 时 间 以 后 ， 对 象 可 能 被 销毁 ， 但 地 盘 必 须 加 以 重用 。 在 某 些 Java 虚 拟 机 中 ， 堆 的 实现 截然 
不 同 ， 它 更 像 一 个 传送 带 ， 每 分 配 一 个 新 对 象 ， 它 就 往 前 移动 一 格 。 这 意味 着 对 象 存储 空间 的 
分 配 速度 非常 决 。Java 的 “ 堆 指针 ”只 是 简单 地 移动 到 尚未 分 配 的 区 域 ， 其 效率 比 得 上 C++ 在 堆 
栈 上 分 配 空间 的 效率 。 当 然 ， 实 际 过 程 中 在 敌 记 工作 方面 还 有 少量 额外 开销 ， 但 比 不 上 查找 可 
用 空间 开销 大 。 

读者 也 许 已 经 意识 到 了 ，Java 中 的 堆 未 必 完 全 像 传 送 带 那样 工作 。 要 真是 那样 的 话 ， 势 必 
会 导致 频繁 的 内 存 页 面 调度 一 将 其 移 进 移出 硬盘 ， 因 此 会 显得 需要 拥有 比 实际 需要 更 多 的 内 
存 。 页 面 调度 会 显著 地 影响 性 能 ， 最 终 ， 在 创建 了 足够 多 的 对 象 之 后 ， 内 存 资源 将 耗 尽 。 其 中 
的 秘密 在 于 垃圾 回收 器 的 介入 。 当 它 工作 时 ， 将 一 面 回收 空间 ， 一 面 使 堆 中 的 对 象 紧凑 排列 ， 
这 样 “ 堆 指针 ”就 可 以 很 容易 移动 到 更 靠近 传送 带 的 开始 处 ， 也 就 尽量 避免 了 页 面 错误 。 通 过 
垃圾 回收 器 对 对 象 重新 排列 ， 实 现 了 一 种 高 速 的 、 有 无 限 空间 可 供 分 配 的 堆 模型 。 

要 想 更 好 地 理解 Java 中 的 垃圾 回收 ， 先 了 解 其 他 系统 中 的 垃圾 回收 机 制 将 会 很 有 帮助 。 引 
用 记 数 是 一 种 简单 但 速度 很 慢 的 垃圾 回收 技术 。 每 个 对 象 都 含有 一 个 引用 记 数 器 ， 当 有 引用 连 
接 至 对 象 时 ， 引 用 计数 加 1。 当 引用 离开 作用 域 或 被 置 为 null 时 ， 引 用 计数 减 1。 虽 然 管理 引用 记 
数 的 开销 不 大 ， 但 这 项 开销 在 整个 程序 生命 周期 中 将 持续 发 生 。 垃 圾 回收 器 会 在 含有 全 部 对 象 
的 列表 上 遍历 ， 当 发 现 某 个 对 象 的 引用 计数 为 0 时 ， 就 释放 其 占用 的 空间 〈 但 是 ， 引 用 记 数 模式 
经 常会 在 记 数值 变 为 0 时 立即 释放 对 象 )。 这 种 方法 有 个 缺陷 ， 如 果 对 象 之 间 存 在 循环 引用 ， 可 
能 会 出 现 “对 象 应 该 被 回收 ， 但 引用 计数 却 不 为 零 ”的 情况 。 对 垃圾 回收 器 而 言 ， 定 位 这 样 的 
交互 自 引用 的 对 象 组 所 需 的 工作 量 极 大 。 引 用 记 数 常用 来 说 明 垃 圾 收集 的 工作 方式 ， 但 似乎 从 
未 被 应 用 于 任何 一 种 Java 虚 拟 机 实现 中 。 

在 一 些 更 快 的 模式 中 ， 垃 圾 回收 器 并 非 基 于 引用 记 数 技术 。 它 们 依据 的 思想 是 ， 对 任何 
“ 活 ”的 对 象 ， 一 定 能 最 终 追 溯 到 其 存活 在 堆栈 或 静态 存储 区 之 中 的 引用 。 这 个 引用 链条 可 能 会 
穿 过 数 个 对 象 层次 。 由 此 ， 如 果 从 堆栈 和 静态 存储 区 开始 ， 遍 历 所 有 的 引用 ， 就 能 找到 所 有 
“ 活 ” 的 对 象 。 对 于 发 现 的 每 个 引用 ， 必 须 追踪 它 所 引用 的 对 象 ， 然 后 是 此 对 象 包含 的 所 有 引用 
如 此 反复 进行 ， 直 到 “根源 于 堆栈 和 静态 存储 区 的 引用 ”所 形成 的 网 络 全 部 被 访问 为 止 。 你 所 
访问 过 的 对 象 必须 都 是 “ 活 ” 的 。 注 意 ， 这 就 解决 了 “交互 自 引用 的 对 象 组 ”的 问题 一 -这 种 
现象 根本 不 会 被 发 现 ， 因 此 也 就 被 自动 回收 了 。 

在 这 种 方式 下 ，Java 虚 拟 机 将 采用 一 种 自 志 应 的 垃圾 回收 技术 。 至 于 如 何 处 理 找到 的 存活 
对 象 ， 取 决 于 不 同 的 Java 虚 拟 机 实现 。 有 一 种 做 法 名 为 停止 复制 (stop-and-copy)。 显 然 这 意味 
着 ， 先 暂停 程序 的 运行 (所 以 它 不 属于 后 台 回收 模式 ) ， 然 后 将 所 有 存活 的 对 象 从 当前 堆 复制 到 
另 一 个 堆 ， 没 有 被 复制 的 全 部 都 是 垃圾 。 当 对 象 被 复制 到 新 堆 时 ， 它 们 是 一 个 挨 着 一 个 的 ， 所 
以 新 堆 保持 紧凑 排列 ， 然 后 就 可 以 按 前 述 方法 简单 、 直 接地 分 配 新 空间 了 

当 把 对 象 从 一 处 撒 到 另 一 处 时 ， 所 有 指向 它 的 那些 引用 都 必须 修正 。 位 于 堆 或 静态 存储 区 
的 引用 可 以 直接 被 修正 ， 但 可 能 还 有 其 他 指向 这 些 对 象 的 引用 ， 它 们 在 遍历 的 过 程 中 才能 被 找 
到 (可 以 想像 成 有 个 表格 ， 将 旧地 址 映射 至 新 地 址 ) 。 

对 于 这 种 所 谓 的 “复制 式 回收 器 ”而 言 ， 效 率 会 降低 ， 这 有 两 个 原因 。 首 先 ， 得 有 两 个 堆 ， 
然后 得 在 这 两 个 分 离 的 堆 之 间 来 回 的 腾 ， 从 而 得 维护 比 实际 需要 多 一 倍 的 空间 。 某 些 Java 虚 拟 
机 对 此 问题 的 处 理 方式 是 ， 按 需 从 堆 中 分 配 几 块 较 大 的 内 存 ， 复 制 动 作 发 生 在 这 些 大 块 内 存 
之 间 。 

第 二 个 问题 在 于 复制 。 程 序 进入 稳定 状态 之 后 ， 可 能 只 会 产生 少量 垃圾 ， 甚 至 没有 垃圾 。 
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尽管 如 此 ， 复 制式 回收 器 仍然 会 将 所 有 内 存 自 一 处 复制 到 另 一 处 ， 这 很 浪费 。 为 了 避免 这 种 情 
形 ， 一 些 Java 虚 拟 机 会 进行 检查 : 要 是 没有 新 垃圾 产生 ， 就 会 转换 到 另 一 种 工作 模式 ( 即 “ 自 
适应 ”)。 这 种 模式 称 为 标记 一 清扫 (mark-and-sweep)，Sun 公 司 早期 版 本 的 Java 虚 拟 机 使 用 了 这 
种 技术 。 对 一 般 用 途 而 言 ,“ 标 记 - 清 扫 ” 方 式 速度 相当 慢 ， 但 是 当 你 知道 只 会 产生 少量 垃圾 甚 
至 不 会 产生 垃圾 时 ， 它 的 速度 就 很 快 了 。 

“标记 一 清扫 ”所 依据 的 思路 同样 是 从 堆栈 和 静态 存储 区 出 发 ， 遍 历 所 有 的 引用 ， 进 而 找 出 
所 有 存活 的 对 象 。 每 当 它 找到 一 个 存活 对 象 ， 就 会 给 对 象 设 一 个 标记 ， 这 个 过 程 中 不 会 回收 任 
何 对 象 。 只 有 全 部 标记 工作 完成 的 时 候 ， 清 理 动 作 才 会 开始 。 在 清理 过 程 中 ， 没 有 标记 的 对 象 
将 被 释放 ， 不 会 发 生 任何 复制 动作 。 所 以 剩 下 的 堆 空间 是 不 连续 的 ， 垃 圾 回收 器 要 是 希望 得 到 
连续 空间 的 话 ， 就 得 重新 整理 剩 下 的 对 象 。 

“停止 ~ 复制 ”的 意思 是 这 种 垃圾 回收 动作 不 是 在 后 台 进 行 的 ， 相 反 ， 垃 圾 回收 动作 发 生 的 
同时 ， 程 序 将 会 被 暂停 。 在 Sun 公司 的 文档 中 会 发 现 ， 许 多 参考 文献 将 垃圾 回收 视 为 低 优先 级 
的 后 台 进 程 ， 但 事实 上 垃圾 回收 器 在 Sun 公 司 早期 版 本 的 Java 虚 拟 机 中 并 非 以 这 种 方式 实现 的 。 
当 可 用 内 存 数量 较 低 时 ，Sun 版 本 的 垃圾 回收 器 会 暂停 运行 程序 ， 同 样 , “标记 -清扫 ”工作 也 
必须 在 程序 暂停 的 情况 下 才能 进行 。 

如 前 文 所 述 ， 在 这 里 所 讨论 的 Java 虚 拟 机 中 ， 内 存 分 配 以 较 大 的 “ 块 ”为 单位 。 如 果 对 象 
较 大 ， 它 会 占用 单独 的 块 。 严 格 来 说 ,“ 停 止 -复制 ”要 求 在 释放 旧 有 对 象 之 前 ， 必 须 先 把 所 有 
存活 对 象 从 旧 堆 复制 到 新 堆 ， 这 将 导致 大 量 内 存 复制 行为 。 有 了 块 之 后 ， 垃 圾 回收 器 在 回收 的 
时 候 就 可 以 往 废弃 的 块 里 拷贝 对 象 了 。 每 个 块 都 用 相应 的 代数 (generation count) 来 记录 它 是 否 
还 存活 。 通 常 ， 如 果 块 在 某 处 被 引用 ， 其 代数 会 增加 ， 垃 圾 回收 器 将 对 上 次 回收 动作 之 后 新 分 
配 的 块 进行 整理 。 这 对 处 理 大 量 短命 的 临时 对 象 很 有 帮助 。 垃 圾 回收 器 会 定期 进行 完整 的 清理 
动作 一 一 大 型 对 象 仍然 不 会 被 复制 (只 是 其 代数 会 增加 )， 内 含 小 型 对 象 的 那些 块 则 被 复制 并 整 
理 。Java 虚 拟 机 会 进行 监视 ， 如 果 所 有 对 象 都 很 稳定 ， 垃 圾 回收 器 的 效率 降低 的 话 ， 就 切换 到 
“标记 - 清扫” 方式， 同样，Java 虚 拆 机 会 跟踪 “标记 -清扫 ”的 效果 ， 要 是 堆 空间 出 现 很 多 碎片 ， 
就 会 切换 回 “停止 -复制 ”方式 。 这 就 是 “ 自 适 应 ”技术 ， 你 可 以 给 它 个 罗 嗪 的 称呼 :“ 自 适应 
的 、 分 代 的 、 停 止 -~ 复制、 标记- 清扫 ” 式 垃圾 回收 器 。 

Java 虚 拟 机 中 有 许多 附加 技术 用 以 提升 速度 。 尤 其 是 与 加 载 器 操作 有 关 的 ， 被 称 为 “即时 ” 
(Just-In-Time, JIT) 编译 器 的 技术 。 这 种 技术 可 以 把 程序 全 部 或 部 分 翻译 成 本 地 机 器 码 (这 
本 来 是 Java 虚 拟 机 的 工作 ) ， 程 序 运行 速度 因此 得 以 提升 。 当 需要 装载 某 个 类 (通常 是 在 为 该 
类 创建 第 一 个 对 象 ) 时， 编译 器 会 先 找到 其 class 文件 ， 然 后 将 该 类 的 字 节 码 装 入 内存。 此 时 ， 
有 两 种 方案 可 供 选 择 。 一 种 是 就 让 即时 编译 器 编译 所 有 代码 。 但 这 种 做 法 有 两 个 缺陷 ; 这 种 
加 载 动作 散落 在 整个 程序 生命 周期 内 ， 累 加 起 来 要 花 更 多 时 间 ， 并 且 会 增加 可 执行 代码 的 长 
度 〈 字 节 码 要 比 即时 编译 器 展开 后 的 本 地 机 器 码 小 很 多 ) ， 这 将 导致 页 面 调度 ， 从 而 降低 程序 
速度 。 另 一 种 做 法 称 为 惰性 评估 (lazy evaluation) ， 意 思 是 即时 编译 器 只 在 必要 的 时 候 才 编 译 
代码 。 这 样 ， 从 不 会 被 执行 的 代码 也 许 就 压根 不 会 被 JIT 所 编译 。 新 版 JDK 中 的 Java HotSpot 技 
术 就 采用 了 类 似 方法 ， 代 码 每 次 被 执行 的 时 候 都 会 做 一 些 优化 ， 所 以 执行 的 次 数 越 多 ， 它 的 
速度 就 越 快 。 


5.6 成 员 初始 化 


Java 尽 力 保证 : 所 有 变量 在 使 用 前 都 能 得 到 恰当 的 初始 化 。 对 于 方法 的 局 部 变量 ，Java 以 编 
译 时 错误 的 形式 来 贯彻 这 种 保证 。 所 以 如 果 写 成 : 
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void f() { 
int i; 
i++; // Error -- i not initialized 


就 会 得 到 一 条 出 错 消息 ， 告 诉 你 i 可 能 尚未 初始 化 。 当 然 ， 编 译 器 也 可 以 为 误 一 个 默认 值 ， 但 是 
未 初始 化 的 局 部 变量 更 有 可 能 是 程序 员 的 疏忽 ， 所 以 采用 默认 值 反而 会 掩盖 这 种 失误 。 因 此 强 
制程 序 员 提供 一 个 初始 值 ， 往 往 能 够 帮助 找 出 程序 里 的 缺陷 。 

要 是 类 的 数据 成 员 ( 即 字段 ) 是 基本 类 型 ， 情 况 就 会 变 得 有 些 不 同 。 正 如 在 “一 切 都 是 对 
象 ”一 章 中 所 看 到 的 ， 类 的 每 个 基本 类 型 数据 成 员 保证 都 会 有 一 个 初始 值 。 下 面 的 程序 可 以 验 
证 这 类 情况 ， 并 显示 它们 的 值 : 


1/: initialization/InitialValues. java 
// Shows default initial values. 
import static net.mindview.util.Print.*; 


public class InitialValues { 
boolean t; 
char c; 
byte b; 
short s; 
int 4; 
long 1; 
float f; 
double d; 
InitialValues reference; 
void printInitialvalues() { 


print("Data type Initial value"); 
print("boolean "+0: 
print("char eee in): 
print ("byte "+ b); 
print("short +s); 
print("int "4d: 
print("long "+0; 
print("float "+t: 
print("double "+ dd); 
print("reference ”+ reference); 


} 

public static void main(String{] args) { 
InitialValues iv = new InitialValues(); 
iv. printInitialvalues(); 
/* You could also say: 
new InitialValues() .printinitialValues() ; 


对 
} 

} /* Output: 
Data type Initial value 
boolean false 
char 1] 
byte 9 
short 9 
int e 
tong 9 
float 6.9 
double 0.0 
reference null 
Whim 


可 见 尽管 数据 成 员 的 初 值 没有 给 出 ， 但 它们 确实 有 初 值 (char 值 为 0， 所 以 显示 为 空白 ) 。 
这 样 至 少 不 会 冒 “ 未 初始 化 变量 ”的 风险 了 。 
在 类 里 定义 一 个 对 象 引用 时 ， 如 果 不 将 其 初始 化 ， 此 引用 就 会 获得 一 个 特殊 值 null。 


上 
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5.6.1 指定 初始 化 

如 果 想 为 某 个 变量 赋 初 值 ， 该 怎么 做 呢 ? 有 一 种 很 直接 的 办 法 ， 就 是 在 定义 类 成 员 变 量 的 
地 方 为 其 赋值 (注意 在 C++ 里 不 能 这 样 做 ， 尽 管 C++ 的 新 手 们 总 想 这 样 做 ) 。 以 下 代码 片段 修改 
了 InitialValues 类 成 员 变量 的 定义 ， 直 接 提供 了 初 值 。 


/1/: initialization/InitialValues2. java 
// Providing explicit initial values. 


public class InitialValues2 { 
boolean bool = true; 
char ch = 'x'; 





} Mi~ 


也 可 以 用 同样 的 方法 初始 化 非 基本 类 型 的 对 象 。 如 果 Depth 是 一 个 类 ， 那 么 可 以 像 下 面 这 样 
创建 一 个 对 象 并 初始 化 它 : 

/1/: initialization/Measurement. java 

class Depth {} 


public class Measurement { 
Depth d = new Depth(); 
ge 

} :~ 


如 果 没 有 为 a 指 定 初始 值 就 尝试 使 用 它 ， 就 会 出 现 运行 时 错误 ， 告 诉 你 产生 了 一 个 异常 (这 
在 第 12 章 中 详 述 ) 。 
其 至 可 以 通过 调用 某 个 方法 来 提供 初 值 : 


//: initialization/MethodInit. java 
public class MethodInit { 


int i = fO; 
int fO { return 11; } 
} Mi~ 


这 个 方法 也 可 以 带 有 参数 ， 但 这 些 参数 必须 是 已 经 被 初始 化 了 的 。 因 此 ， 可 以 这 样 写 : 


//: initiatization/MethodInit2.java 
public class MethodInit2 { 

int 1 = fO; 

int j = gG); 

int fO { return 11; } 

int g(int n) { return n * 10; } 
} Ii~ 


但 像 下 面 这 样 写 就 不 对 了 : 


/1: initialization/MethodInit3. java 
public class MethodInit3 { 
//! int j = g(i); // Illegal forward reference 
int i = fO; 
int fO { return 11; } 
int glint n) { return n * 10; } 
} i~ 


显然 ， 上 述 程序 的 正确 性 取决 于 初始 化 的 顺序 ， 而 与 其 编译 方式 无 关 。 所 以 ， 编 译 器 恰当 地 对 
“向 前 引用 ”发 出 了 警告 。 
这 种 初始 化 方法 既 简单 又 直观 。 但 有 个 限制 : 类 InitialValues 的 每 个 对 象 都 会 具有 相同 的 初 
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值 。 有 时 ， 这 正 是 所 希望 的 ， 但 有 时 却 需要 更 大 的 灵活 性 。 
5.7 构造 器 初始 化 


可 以 用 构造 器 来 进行 初始 化 。 在 运行 时 刻 ， 可 以 调用 方法 或 执行 某 些 动作 来 确定 初 值 ， 这 
为 编程 带 来 了 更 大 的 灵活 性 。 但 要 牢记 : 无 法 阻止 自动 初始 化 的 进行 ， 它 将 在 构造 器 被 调用 之 
前 发 生 。 因 此 ， 假 如 使 用 下 述 代码 : 


11; initialization/Counter .java 
public class Counter { 

int i; 

Counter() (i = 7: } 

Vises 
} Mi~ 


那么 i 首 先 会 被 置 0， 然 后 变 成 ?7。 对 于 所 有 基本 类 型 和 对 象 引用 ， 包 括 在 定义 时 已 经 指定 初 
值 的 变量 ， 这 种 情况 都 是 成 立 的 ， 因 此 ， 编 译 器 不 会 强制 你 一 定 要 在 构造 器 的 某 个 地 方 或 在 使 
用 它们 之 前 对 元 素 进行 初始 化 一 一 因为 初始 化 早已 得 到 了 保证 。 


5.7.1 初始 化 顺序 
在 类 的 内 部 ， 变 量 定义 的 先后 顺序 决定 了 初始 化 的 顺序 。 即 使 变量 定义 散布 于 方法 定义 之 
间 ， 它 们 仍旧 会 在 任何 方法 〈 包 括 构造 器 ) 被 调用 之 前 得 到 初始 化 。 例 如 : 


//; initialization/OrderOfInitialization. java 
// Demonstrates initialization order. 
import static net.mindview.util.Print.*; 


// When the constructor is called to create a 
// Window object, you'll see a message: 
class Window { 

Window(int marker) { print("Window(" + marker + ")"); } 
$ 


class House { 
Window w1 = new Window(1); // Before constructor 
House() { 
// Show that we're in the constructor: 
print("House()") ; 
W3 = new Window(33); // Reinitialize w3 


Window w2 = new Window(2); // After constructor 
void FO) { print(*f()"); } 
Window w3 = new Window(3); // At end 

} 


public class OrderOfInitialization { 
public static void main(String[] args) { 
House h = new House(); 
h.f Qi // Shows that construction is done 


} 
} /* Output: 
Window (1) 
Window (2) 
Window(3) 
House() 
Window (33) 
t0 
111 :~ 


在 House 类 中 ， 故 意 把 几 个 Window 对 象 的 定义 散布 到 各 处 ， 以 证 明 它们 全 都 会 在 调用 构造 
器 或 其 他 方法 之 前 得 到 初始 化 。 此 外 ，w3 在 构造 器 内 再 次 被 初始 化 。 
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由 输出 可 见 ，w3 这 个 引用 会 被 初始 化 两 次 : 一 次 在 调用 构造 器 前 ， 一 次 在 调用 期 间 (第 一 
次 引用 的 对 象 将 被 丢弃 ， 并 作为 垃圾 回收 )。 试 想 ， 如 果 定义 了 一 个 重 载 构造 器 ， 它 没有 初始 化 
w3, 同时 在 w3 的 定义 里 也 没有 指定 默认 值 ， 那 会 产生 什么 后 果 呢 ? 所 以 尽管 这 种 方法 似乎 效率 
不 高 ， 但 它 的 确 能 使 初始 化 得 到 保证 。 

5.7.2 静态 数据 的 初始 化 

无 论 创建 多 少 个 对 象 ， 静 态 数据 都 只 占用 一 份 存储 区 域 。static 关 键 字 不 能 应 用 于 局 部 变量 ， 
因此 它 只 能 作用 于 域 。 如 果 一 个 域 是 静态 的 基本 类 型 域 ， 且 也 没有 对 它 进行 初始 化 ， 那 么 它 就 
会 获得 基本 类 型 的 标准 初 值 ， 如 果 它 是 一 个 对 象 引用 ， 那 么 它 的 默认 初始 化 值 就 是 null。 

如 果 想 在 定义 处 进行 初始 化 ， 采 取 的 方法 与 非 静态 数据 没什么 不 同 。 

要 想 了 解 静态 存储 区 域 是 何 时 初始 化 的 ， 就 请 看 下 面 这 个 例子 : 


/11: inittalization/StaticInitialization. java 
// Specifying initial values in a class definition. 
import static net.mindview.util.Print.*; 


class Bowl { 
Bowl(int marker) { 
print("Bowl(" + marker + ")"); 


} 
void fl(int marker) { 
print("f1(" + marker + ")"); 
) 
) 


class Table { 
static Bowl bowll = new Bowl(1); 
Table() { 
print(*Table()"): 
bowl2.f1(1); 


} 
void f2(int marker) { 
print("f2(" + marker + ")"); 


} 
static Bowl bowl2 = new Bowl(2); 
} 


class Cupboard { 
Bowl bowl3 = new-Bowl (3); 
static Bowl bowl4 = new Bowl(4); 
Cupboard() { 
print ("Cupboard()"); 
bowl4.f1(2); 


void f3(int marker) { 
print("f3(" + marker + ")"); 


} 
static Bowl bow15 = new Bowl(5); 
} 


public class StaticInitialization { 
public static void main(String[] args) { 
print("Creating new Cupboard() in main”); 
new Cupboard(); 
print("Creating new Cupboard() in main"); 
new Cupboard(); 
table. f2(1); 
cupboard. f3(1): 
} 
static Table table = new Table(); 
static Cupboard cupboard = new Cupboard(): 
} /* Output: 
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Bowl (1) 
Bowl (2) 

Table() 

na) 

Bowl (4) 

Bowl (5) 

Bowl (3) 

Cupboard () 

1(2) 

Creating new Cupboard() in main 
Bowl (3) 

Cupboard() 

f1(2) 

Creating new Cupboard() in main 
Bowl (3) 

Cupboard() 

f1(2) 

na) 

f3(1) 

“Ii~ 


Bowl 类 使 得 看 到 类 的 创建 ， 而 Table 类 和 Cupboard 类 在 它们 的 类 定义 中 加 入 了 Bowl 类 型 的 
静态 数据 成 员 。 注 意 ， 在 静态 数据 成 员 定义 之 前 ，Cupboard 类 先 定 义 了 一 个 Bowl 类 型 的 非 静 坊 
数据 成 员 b3。 

由 输出 可 见 ， 静 态 初始 化 只 有 在 必要 时 刻 才 会 进行 。 如 果 不 创建 Table 对 象 ， 也 不 引用 
Table.b1 或 Table.b2， 那 么 静态 的 Bowl bl 和 b2 永 远 都 不 会 被 创建 。 只 有 在 第 一 个 Table 对 象 被 
创建 (或 者 第 一 次 访问 静态 数据 ) 的 时 候 ， 它 们 才 会 被 初始 化 。 此 后 ， 静态 对 象 不 会 再 次 被 初 
始 化 。 

初始 化 的 顺序 是 先 静 态 对 象 (如 果 它 们 尚未 因 前 面 的 对 象 创建 过 程 而 被 初始 化 ) ， 而 后 是 
“ 非 静态 ”对 象 。 从 输出 结果 中 可 以 观察 到 这 一 点 。 要 执行 main() (静态 方法 ) ， 必 须 加 载 
StaticInitialization 类 ， 然 后 其 静态 域 table 和 cupboard 被 初始 化 ， 这 将 导致 它们 对 应 的 类 也 被 加 
载 ， 并 且 由 于 它们 也 都 包含 静态 的 Bowl 对 象 ， 因 此 Bowl 随 后 也 被 加 载 。 这 样 ， 在 这 个 特殊 的 程 
序 中 的 所 有 类 在 main0 开 始 之 前 就 都 被 加 载 了 。 实 际 情况 通常 并 非 如 此 ， 因 为 在 典型 的 程序 中 ， 
不 会 像 在 本 例 中 所 做 的 那样 ， 将 所 有 的 事物 都 通过 static 联 系 起 来 。 

总 结 一 下 对 象 的 创建 过 程 ， 假 设 有 个 名 为 Dog 的 类 : 

1. 即使 没有 显 式 地 使 用 static 关 键 字 ， 构 造 器 实际 上 也 是 静态 方法 。 因 此 ， 当 首次 创建 类 型 
为 Dog 的 对 象 时 〈 构 造 器 可 以 看 成 静态 方法 ) ， 或 者 Dog 类 的 静态 方法 /静态 域 首次 被 访问 时 ， 
Java 解 释 器 必须 查找 类 路 径 ， 以 定位 Dog.class 文 件 。 

2. 然后 载 人 Dog.class (后 面 会 学 到 ， 这 将 创建 一 个 Class 对 象 )， 有 关 静 态 初始 化 的 所 有 动作 
都 会 执行 。 因 此 ， 静 态 初始 化 只 在 Class 对 象 首次 加 载 的 时 候 进 行 一 次 。 

3. 当 用 new Dog0 创 建 对 象 的 时 候 ， 首 先 将 在 堆 上 为 Dog 对 象 分 配 足够 的 存储 空间 。 

4. 这 块 存储 空间 会 被 清 零 ， 这 就 自动 地 将 Dog 对 象 中 的 所 有 基本 类 型 数据 都 设置 成 了 默认 值 
(对 数字 来 说 就 是 0， 对 布尔 型 和 字符 型 也 相同 ) ， 而 引用 则 被 设置 成 了 null。 

5. 执行 所 有 出 现 于 字段 定义 处 的 初始 化 动作 。 

6. 执行 构造 器 。 正 如 将 在 第 7 章 所 看 到 的 ， 这 可 能 会 牵涉 到 很 多 动作 ， 尤 其 是 涉及 继承 的 
时 候 。 

5.7.3 显 式 的 静态 初始 化 

Java 允 许 将 多 个 静态 初始 化 动作 组 织 成 一 个 特殊 的 “静态 子 句 ”( 有 了 时 也 叫做 “静态 块 ")。 

就 像 下 面 这 样 : 
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/1/: initialization/Spoon. java 
public class Spoon { 
static int 1; 
static { 
1 = 47; 


} 
Mix 
尽管 上 面 的 代码 看 起 来 像 个 方法 ， 但 它 实际 只 是 一 段 跟 在 static 关 键 字 后 面 的 代码 。 与 其 他 
静态 初始 化 动作 一 样 ， 这 段 代码 仅 执行 一 次 : 当 首 次 生成 这 个 类 的 一 个 对 象 时 ， 或 者 首次 访问 
属于 那个 类 的 静态 数据 成 员 时 (即便 从 未 生成 过 那个 类 的 对 象 )。 例 如 : 


1/: initialization/ExplicitStatic. java 
// Explicit static initialization with the "static" clause. 
import static net.mindview.util.Print.*; 


class Cup { 
Cup(int marker) { 
print("Cup(" + marker + *)"); 


} 
void f(int marker) { 
print("f(" + marker + ")"); 
4 
} 


class Cups { 
static Cup cupl; 
static Cup cup2; 
static { 
cupl = new Cup(1); 
cup2 = new Cup(2); 
} 
Cups() { 
print ("Cups()"); 
t 
) 
public class ExplicitStatic { d 
public static void main(String[] args) { 
print("Inside main()"); 
Cups.cupl.f(99); // (1) 


// static Cups cupsl = new Cups(); // (2) 
11 static Cups cups2 = new Cups(): // (2) 

} /* Output: 

Inside main() 

Cup(1) 

Cup(2) 

f(99) 

Wh 


无 论 是 通过 标 为 (0 的 那 行 代码 访问 静态 的 cup1 对 象 ， 还 是 把 标 为 (0) 的 行 注释 掉 ， 让 它 去 运行 
标 为 (2) 的 那 行 代码 〈 即 解除 标 为 (2) 的 行 的 注释 ) ，Cups 的 静态 初始 化 动作 都 会 得 到 执行 。 如 果 把 
标 为 (0 和 (2) 的 行 同时 注释 掉 ，Cups 的 静态 初始 化 动作 就 不 会 进行 ， 就 像 在 输出 中 看 到 的 那样 。 此 
外 ， 激 活 一 行 还 是 两 行 标 为 (2) 的 代码 ( 即 解除 注释 ) 都 无 关 紧要 ， 静 态 初始 化 动作 只 进行 一 次 。 

练习 13: (1) 验证 前 面 段落 中 的 语句 。 

练习 14: (1) 编写 一 个 类 ， 拥 有 两 个 静态 字符 串 域 ， 其 中 一 个 在 定义 处 初始 化 ， 另 一 个 在 静 
态 块 中 初始 化 。 现 在 ， 加 入 一 个 静态 方法 用 以 打印 出 两 个 字段 值 。 请 证 明 它们 都 会 在 被 使 用 之 
前 完成 初始 化 动作 。 

5.7.4 非 静态 实例 初始 化 
Java 中 也 有 被 称 为 实例 初始 化 的 类 似 语法 ， 用 来 初始 化 每 一 个 对 象 的 非 静态 变量 。 例 如 : 











191 











192) 











98 第 5 新 





/1/: imitialization/Mugs. java 
// Java “Instance Initialization.” 
import static net.mindview.util.Print.*; 


class Mug { 
Mug(int marker) { 
print(*Mug(" + marker + “)"): 


} 
void f(int marker) { 
print("f(" + marker + ")"); 


} 
public class Hugs { 
Mug mugi: 
Mug mug2; 
{ 
mugi = new Mug(1); 
mug2 = new Mug(2) ; 
print("mugl & mug2 initialized"); 
} 
Mugs() { 
print("Mugs()"); 


} 
Mugs(int i) { 
print("Mugs(int)"); 


} 
public static void main(String[] args) { 
print("Inside main()"); 
new Mugs(); 
print("new Mugs() completed"); 
new Mugs (1); 
print("new Mugs(1) completed") ; 
} 
} /* Output: 
Inside main() 
Mug (1) 
Mug (2) 
mugl & mug2 initialized 
Mugs () 
new Mugs() completed 
Mug(1) 
Mug(2) 
mugl & mug2 initialized 
Mugs(int) 
new Mugs(1) completed 
Wham 


你 可 以 看 到 实例 初始 化 子 句 : 
mugl = new Mug(1); 
mug2 = new Mug(2); 
print("mugl & mug2 initialized"); 


看 起 来 它 与 静态 初始 化 子 名 一模一样 ， 只 不 过 少 了 static 关 键 字 。 这 种 语法 对 于 支持 “匿名 
内 部 类 ”( 参 见 第 10 章 ) 的 初始 化 是 必须 的 ， 但 是 它 也 使 得 你 可 以 保证 无 论调 用 了 哪个 显 式 构造 
器 ， 某 些 操作 都 会 发 生 。 从 输出 中 可 以 看 到 实例 初始 化 子 句 是 在 两 个 构造 器 之 前 执行 的 。 

练习 15: (1) 编写 一 个 含有 字符 串 域 的 类 ， 并 采用 实例 初始 化 方式 进行 初始 化 。 


5.8 数组 初始 化 


数组 只 是 相同 类 型 的 、 用 一 个 标识 符 名 称 封装 到 一 起 的 一 个 对 象 序列 或 基本 类 型 数据 序列 。 
数组 是 通过 方 括号 下 标 操作 符 [ ] 来 定义 和 使 用 的 。 要 定义 一 个 数组 ， 只 需 在 类 型 名 后 加 上 一 对 
空 方 括号 即 可 : 


| a 
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inti al 

方 括号 也 可 以 置 于 标识 符 后 面 : 

int alt}: 

两 种 格式 的 含义 是 一 样 的 ， 后 一 种 格式 符合 C 和 C++ 程 序 员 的 习惯 。 不 过 ， 前 一 种 格式 或 许 
更 合理 ， 毕 竞 它 表明 类 型 是 “一 个 int 型 数组 "。 本 书 将 采用 这 种 格式 。 

编译 器 不 允许 指定 数组 的 大 小 。 这 就 又 把 我 们 带 回 到 有 关 “ 引 用 ”的 问题 上 。 现 在 拥有 的 
只 是 对 数组 的 一 个 引用 〈 你 已 经 为 该 引用 分 配 了 足够 的 存储 空间 ) ， 而 且 也 没 给 数组 对 象 本 身分 
配 任何 空间 。 为 了 给 数组 创建 相应 的 存储 空间 ， 必 须 写 初始 化 表达 式 。 对 于 数组 ， 初 始 化 动作 
可 以 出 现在 代码 的 任何 地 方 ， 但 也 可 以 使 用 一 种 特殊 的 初始 化 表达 式 ， 它 必须 在 创建 数组 的 地 
方 出 现 。 这 种 特殊 的 初始 化 是 由 一 对 花 括号 括 起 来 的 值 组 成 的 。 在 这 种 情况 下 ， 存 储 空间 的 分 
配 (等 价 于 使 用 new) 将 由 编译 器 负责 。 例 如 


inti] al = { 1, 2,3, 4,55 

那么 ， 为 什么 还 要 在 没有 数组 的 时 候 定义 一 个 数组 引用 呢 ? 
int[] a2; 

在 Java 中 可 以 将 一 个 数组 赋值 给 另 一 个 数组 ， 所 以 可 以 这 样 : 
a2 = al 


其 实 真正 做 的 只 是 复制 了 一 个 引用 ， 就 像 下 面 演示 的 那样: 


//: initialization/ArraysOfPrimitives.java 
import static net.mindview.util.Print.*; 


public class ArraysOfPrimitives { 
public static void main(String args) { 

int{] al = { 1, 2, 3, 4,5) 

int[] a2; 

a2 = al; 

for(int i = 0; 1 < a2.length; i++) 
a2ti} = a2ti) + 1; 

for(int 1 = 0; i < al.length; i++) 
print("al(" + 1 + "] =" + al[li)); 


} /* Output: 

al[6] = 2 

al[1] = 3 

al[2] = 4 

al[3] = 5 

al[4] = 6 

Whim 

可 以 看 到 代码 中 给 出 了 ai 的 初始 值 ， 但 a2 却 没有 ， 在 本 例 中 ，a2 是 在 后 面 被 赋 给 另 一 个 数 
组 的 。 由 于 a2 和 al 是 相同 数组 的 别名 ， 因 此 通过 a2 所 做 的 修改 在 al 中 可 以 看 到 。 

所 有 数组 (无论 它们 的 元 素 是 对 象 还 是 基本 类 型 ) 都 有 一 个 固有 成 员 ， 可 以 通过 它 获知 数 
组 内 包含 了 多 少 个 元 素 ， 但 不 能 对 其 修改 。 这 个 成 员 就 是 length。 与 C 和 C++ 类 似 ，Java 数 组 计 
数 也 是 从 第 0 个 元 素 开 始 ， 所 以 能 使 用 的 最 大 下 标 数 是 length-1。 要 是 超出 这 个 边界 ，C 和 C++ 会 
“默默 ”地 接受 ， 并 允许 你 访问 所 有 内 存 ， 许 多 声名 狼藉 的 程序 错误 由 此 而 生 。Java 则 能 保护 你 
免 受 这 一 问题 的 困扰 ， 一 旦 访问 下 标 过 界 ， 就 会 出 现 运行 时 错误 〈 即 异常 ) 。 。 


O 当然 ， 每 次 访问 数组 的 时 候 都 要 检查 边界 的 做 法 在 时 间 和 代码 上 都 是 需要 开销 的 ， 但 是 无 法 禁用 这 个 功能 。 
这 意味 着 如 果 数组 访问 发 生 在 一 些 关 键 节点 上 ， 它 们 有 可 能 会 成 为 导致 程序 效率 低下 的 原因 之 一 。 但 是 基于 
“因特网 的 安全 以 及 提高 程序 员 生产 力 ”的 理由 ，Java 的 设计 者 认为 这 种 权衡 是 值得 的 。 尽 管 你 可 能 会 受到 诱 
惑 ， 去 编写 你 认为 可 以 使 得 数组 访问 效率 提高 的 代码 ， 但 是 这 一 切 都 是 在 浪费 时 间 ， 因 为 自动 的 编译 期 错误 
和 运行 时 优化 都 可 以 提高 数组 访问 的 速度 。 
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如 果 在 编写 程序 时 ， 并 不 能 确定 在 数组 里 需要 多 少 个 元 素 ， 那 么 该 怎么 办 呢 ? 可 以 直接 用 
new 在 数组 里 创建 元 素 。 尽 管 创建 的 是 基本 类 型 数组 ，new 仍 然 可 以 工作 (不 能 用 new 创 建 单个 
的 基本 类 型 数据 ) 。 


/1/: initialization/ArrayNew. java 
// Creating arrays with new. 

import java.util.*; 

import static net.mindview.util.Print.*; 


public class ArrayNew { 
public static void main(String[] args) { 
int[] a; 
Random rand = new Random(47): 
a = new int[rand.nextInt(20)]; 
print("length of a = ”+ a.length); 
print (Arrays. toString(a)) ; 


} hs Output: 

length of a = 18 

19, 6, 0, 9, 0, 0, 0, 0, 6, 0, 9, 0, 9. 6. 6, Ə, Ə, 6] 

Ii~ 

数组 的 大 小 是 通过 Random.nextInt0 方 法 随机 决定 的 ， 这 个 方法 会 返回 0 到 输入 参数 之 间 的 
一 个 值 。 这 表明 数组 的 创建 确实 是 在 运行 时 刻 进行 的 。 此 外 ， 程 序 输出 表明 数组 元 素 中 的 基 
本 数据 类 型 值 会 自动 初始 化 成 空 值 (对 于 数字 和 字符 ， 就 是 0， 对 于 布尔 型 ， 是 false) 。 

Arrays.toString0 方 法 属于 java.util 标 准 类 库 ， 它 将 产生 一 维 数组 的 可 打印 版 本 。 

当然 ， 在 本 例 中 ， 数 组 也 可 以 在 定义 的 同时 进行 初始 化 : 

int[] a = new int[rand.nextInt(20)]; 

如 果 可 能 的 话 ， 应 该 尽量 这 么 做 。 

如 果 你 创建 了 一 个 非 基 本 类 型 的 数组 ， 那 么 你 就 创建 了 一 个 引用 数组 。 以 整 型 的 包装 器 类 
Integer 为 例 ， 它 是 一 个 类 而 不 是 基本 类 型 : 


//: initialization/ArrayClass0bj.java 

// Creating an array of nonprimitive objects. 
import java.util.*; 

import static net.mindview.util.Print.*; 


public class ArrayClassObj { 
public static void main(String[] args) { 
Random rand = new Random(47) ; 
Integer[] a = new Integer {rand.nextInt(29)]; 
print("length of a = " + a.length); 
for(int i = @; i < a.length; i++) 
ali] = rand.nextInt(506); // Autoboxing 
print (Arrays. toString(a)); 


} he Output: (Sample) 

length of a = 18 

(55, 193, 361, 461, 429, 368, 260, 22, 207, 288, 128, 51, 

89, 309, 278, 498, 361, 20] 

Wh 

这 里 ， 即 便 使 用 new 创 建 数组 之 后 : 

Integer[] a = new Integer{rand.nextInt(20)]: 

它 还 只 是 一 个 引用 数组 ， 并 且 直 到 通过 创建 新 的 Integer 对 象 (在 本 例 中 是 通过 自动 包装 机 
制 创建 的 )， 并 把 对 象 赋值 给 引用 ， 初 始 化 进程 才 算 结束 : 


ali] = rand.nextInt (569) ; 


如 果 忘 记 了 创建 对 象 ， 并 且 试 图 使 用 数组 中 的 空 引用 ， 就 会 在 运行 时 产生 异常 。 
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也 可 以 用 花 括 号 括 起 来 的 列表 来 初始 化 对 象 数组 。 有 两 种 形式 


/1: initialization/ArrayInit.java 
11 Array initialization. 
import java.util.*; 


public class ArrayInit { 
public static void main(String[] args) { 
Integer(] a = { 
new Integer (1), 
new Integer (2), 
3, // Autoboxing 


}; 

Integer[] b = new Integer[]{ 
new Integer(1), 

new Integer (2), 

3, // Autoboxing 

}; 
System.out.printtn(Arrays.to5tring(a)); 
System. out.printin(Arrays.toString(b)); 


} 
} /* Output: 
[1, 2, 3) 
(1, 2, 3) 
H~ 


在 这 两 种 形式 中 ， 初 始 化 列表 的 最 后 一 个 逗号 都 是 可 选 的 〈 这 一 特性 使 维护 长 列表 变 得 更 
容易 )。 

尽管 第 一 种 形式 很 有 用 ， 但 是 它 也 更 加 受 限 ， 因 为 它 只 能 用 于 数组 被 定义 之 处 。 你 可 以 在 
任何 地 方 使 用 第 二 种 和 第 三 种 形式 ， 甚 至 是 在 方法 调用 的 内 部 。 例 如 ， 你 可 以 创建 一 个 String 
对 象 数组 ， 将 其 传递 给 另 一 个 main0 方 法 ， 以 提供 参数 ， 用 来 替换 传递 给 该 main0 方 法 的 命令 
行 参数 。 


//: initialization/DynamicArray. java 
1 Array initialization. 


public class DynamicArray { 
public static void main(String{] args) { 
Other.main(new String[]{ "fiddle", "de", “dum” }); 
$ 
} 


class Other { 
public static void main(String[] args) { 


for (String s : args) 
System.out.print(s + " "); 


} ;, Output: 

fiddle de dum 

Whim 

为 Othermain0 的 参数 而 创建 的 数组 是 在 方法 调用 处 创建 的 
可 替换 的 参数 。 

练习 16; (1) 创建 一 个 String 对 象 数据 ， 并 为 每 一 个 元 素 都 赋值 一 个 String。 用 for 循 环 来 打 
印 该 数组 。 

练习 17: (2) 创建 一 个 类 ， 它 有 一 个 接受 一 个 String 参 数 的 构造 器 。 在 构造 阶段 ， 打 印 该 参 
数 。 创 建 一 个 该 类 的 对 象 引用 数组 ， 但 是 不 实际 去 创建 对 象 赋值 给 该 数组 。 当 运行 程序 时 ， 请 
注意 来 自 对 该 构造 器 的 调用 中 的 初始 化 消息 是 否 打印 了 出 来 。 

练习 18: (D 通过 创建 对 象 赋值 给 引用 数组 ， 从 而 完成 前 一 个 练习 。 


因此 你 甚至 可 以 在 调用 时 提供 
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5.8.1 可 变 参 数列 表 

第 二 种 形式 提供 了 一 种 方便 的 语法 来 创建 对 象 并 调用 方法 ， 以 获得 与 C 的 可 变 参 数列 表 
(C 通 常 把 它 简称 为 varargs) 一 样 的 效果 。 这 可 以 应 用 于 参数 个 数 或 类 型 未 知 的 场合 。 由 于 所 有 
的 类 都 直接 或 间接 继承 于 Object 类 ( 随 着 本 书 的 进展 ， 读 者 会 对 此 有 更 深入 的 认识 )， 所 以 可 以 
创建 以 Object 数组 为 参数 的 方法 ， 并 像 下 面 这 样 调用 : 


1/: initialization/VarArgs. java 
// Using array syntax to create variable argument lists. 


class A {} 


public class VarArgs { 
static void printarray(Object(] args) { 
for (Object obj : args) 
System.out.print(obj + " "); 
System.out .printtn(); 
198 } 
public static void main(String[] args) { 
printArray(new Object[]{ 
new Integer(47), new Float(3.14), new Double(11. 11) 
De 


printArray(new Object{]{"one", "two", “three” }); 
printArray(new Object[] {new A(), new A(), new A()}); 














} 
} /* Output: (Sample) 
47 3.14 11.11 
one two three 
A@1a46e30 A@3e25a5 A@19821f 
Whim 


! 可 以 看 到 print0 方 法 使 用 Object 数组 作为 参数 ， 然 后 使 用 foreach 语 法 遍历 数组 ， 打 印 每 个 对 
象 。 标 准 Java 库 中 的 类 能 输出 有 意义 的 内 容 ， 但 这 里 建立 的 类 的 对 象 ， 打 印 出 的 内 容 只 是 类 的 
名 称 以 及 后 面 紧 跟 着 的 一 个 @ 符 号 以 及 多 个 十 六 进 制 数字 。 于 是 ， 默 认 行为 (如 果 没 有 定义 
toString0 方 法 的 话 ， 后 面 会 讲 这 个 方法 的 ) 就 是 打印 类 的 名 字 和 对 象 的 地 址 。 
你 可 能 看 到 过 像 上 面 这 样 编写 的 Java SE5 之 前 的 代码 , 它们 可 以 产生 可 变 的 参数 列表 。 然 而， 
在 Java SE5 中 ， 这 种 盼望 已 久 的 特性 终于 添加 了 进来 ， 因 此 你 现在 可 以 使 用 它们 来 定义 可 变 参 数 
列表 了 ， 就 像 在 printArray0 中 看 到 的 那样 : 


//: initialization/NewVarArgs. java 
/1/ Using array syntax to create variable argument lists. 


public class NewVarArgs { 
static void printarray(Object... args) { 
for (Object obj : args) 
System.out.print(obj + " "): 
System.out.printin(); 


} 
public static void main(String{] args) { 
// Can take individual elements: 
printArray(new Integer(47), new Float(3.14), 
new Double(11.11)); 
printArray(47, 3.14F, 11.11); 
printArray("one", "two", "three"); 
printArray(new A(), new A(). new A(O); 
// Or an array: 
199) printArray((Object[])new Integer[]{ 1, 2, 3, 4 }); 
| printArray(); // Empty list is OK 


} 
} /* Output: (75% match) 
| 47 3.14 11.11 
47 3.14 11.11 
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` one two three 
A@1babS@a ABc3c749 A@156bd4d 
1234 
“Wh 


有 了 可 变 参数 ， 就 再 也 不 用 显 式 地 编写 数组 语法 了 ， 当 你 指定 参数 时 ， 编 译 器 实际 上 会 为 
你 去 填充 数组 。 你 获取 的 仍旧 是 一 个 数组 ， 这 就 是 为 什么 printO 可 以 使 用 foreach 来 迭代 该 数组 
的 原因 。 但 是 ， 这 不 仅仅 只 是 从 元 素 列表 到 数组 的 自动 转换 ， 请 注意 程序 中 倒数 第 二 行 ， 一 个 
Jnteger 数 组 (通过 使 用 自动 包装 而 创建 的 ) 被 转型 为 一 个 Object 数组 (以便 移 除 编译 器 警告 信 
息 )， 并 且 传 递 给 了 printArray0。 很 明显 ， 编 译 器 会 发 现 它 已 经 是 一 个 数组 了 ， 所 以 不 会 在 其 
上 执行 任何 转换 。 因 此 ， 如 果 你 有 一 组 事物 ， 可 以 把 它们 当 作 列表 传递 ， 而 如 果 你 已 经 有 了 一 
个 数组 ， 该 方法 可 以 把 它们 当 作 可 变 参数 列表 来 接受 。 

该 程序 的 最 后 一 行 表明 将 0 个 参数 传递 给 可 变 参数 列表 是 可 行 的 , 当 具 有 可 选 的 尾随 参数 时 ， 
这 一 特性 就 会 很 有 用 : 

/1/: initialization/OptionalTrailingArguments. java 


public class OptionalTrailingArguments { 
static void f(int required, String... trailing) { 
System.out.print("required: ”+ required + * "); 
for(String s : trailing) 
System.out.print(s + " ") 
System.out.printin(); 





} 

public static void main(String{] args) { 
f(1, “one"); 
f(2, "two", "three"); 
(9) 


} 
} /* Output: 
required: 1 one 
required: 2 two three 
required: @ 
Whim 


这 个 程序 还 展示 了 你 可 以 如 何 使 用 具有 Objeet 之 外 类 型 的 可 变 参 数列 表 。 这 里 所 有 的 可 变 
参数 都 必须 是 String 对 象 。 在 可 变 参 数列 表 中 可 以 使 用 任何 类 型 的 参数 ， 包 括 基本 类 型 。 下 面 的 
例子 也 展示 了 可 变 参 数列 表 变 为 数组 的 情形 ， 并 且 如 果 在 该 列表 中 没有 任何 元 素 ， 那 么 转变 成 
的 数据 的 尺寸 为 0: 


//: initialization/VarargType. java 


public class VarargType { 
static void f(Character... args) { 
System. out.print(args.getClass()); 
System.out.printin(" length ”+ args. length); 


} 

static void glint... args) { 
System.out.print(args.getClass()); 
System.out.printin(” length " + args. length); 


} 
public static void main(String(] args) { 
fa"); 


System.out.printin("int{]: "+ new int[@].getClass()): 


} 
} /* Output: 
class [Ljava.lang.Character; length 1 
class [Ljava.lang.Character; length @ 
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class [I length 1 
class [I length © 
int{): class [I 
Wh 


getClass() 方 法 属于 Object 的 一 部 分 ， 我 们 将 在 第 14 章 中 做 全 面 介绍 。 它 将 产生 对 象 的 类 ， 
并 且 在 打印 该 类 时 ， 可 以 看 到 表示 该 类 类 型 的 编码 字符 串 。 前 导 的 “[” 表 示 这 是 一 个 后 面 紧 随 
的 类 型 的 数组 ， 而 紧 随 的 “I” 表 示 基 本 类 型 int。 为 了 进行 双重 检查 ， 我 在 最 后 一 行 创建 了 一 个 
int 数 组 ， 并 打印 了 其 类 型 。 这 样 也 就 验证 了 使 用 可 变 参数 列表 不 依赖 于 自动 包装 机 制 ， 而 实际 
上 使 用 的 是 基本 类 型 。 

然而 ， 可 变 参 数列 表 与 自动 包装 机 制 可 以 和 谐 共处 ， 例 如 : 


11: initiatization/AutoboxingVarargs.java 
public class AutoboxingVarargs { 
public static void f(Integer... args) { 
for(Integer i : args) 
System.out.print(i +" "); 
System. out.printin(); 


} 

public static void main(String{] args) { 
f(new Integer(1), new Integer (2)); 
(4, 5, 6, 7, 8, 9); 
(18, new Integer(11), 12); 


请 注意 ， 你 可 以 在 单一 的 参数 列表 中 将 类 型 混合 在 一 起 ， 而 自动 包装 机 制 将 有 选择 地 将 int 
参数 提升 为 Integer。 
可 变 参 数列 表 使 得 重 载 过 程 变 得 复杂 了 ， 尽 管 乍 一 看 会 显得 足够 安全 


/1: initialization/OverloadingVarargs. java 


‘public class OverloadingVarargs { 
static void f(Character... args) { 
System.out.print(*first"); 
for (Character ¢ : args) 
System.out.print(* " + c); 
System.out.printin(); 


} 
static void f(Integer... args) { 
System.out.print("second"); 
for(Integer i : args) 
System.out.print(" " + i); 
System.out.printin() ; 


} 

static void f(Long... args) { 
System.out.printin("third”) ; 

} 

public static void main(String[] args) { 
fC'at, "bY, e): 
f(D); 
t2. 1): 
f(0); 
(0L); 

| Ji} FO: // Won't compile -- ambiguous 


? 
} /* Output: 
first abe 
second 1 


ke a =n a 
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second 2 1 
second @ 
third 
Whim 


在 每 一 种 情况 中 ， 编 译 器 都 会 使 用 自动 包装 机 制 来 匹配 重 载 的 方法 ， 然 后 调用 最 明确 匹配 
的 方法 。 

但 是 在 不 使 用 参数 调用 f0 时 ， 编 译 器 就 无 法 知道 应 该 调用 哪 一 个 方法 了 。 尽 管 这 个 错误 可 
以 弄 清楚 ， 但 是 它 可 能 会 使 客户 端 程序 员 大 感 意 外 。 

你 可 能 会 通过 在 某 个 方法 中 增加 一 个 非 可 变 参数 来 解决 该 问题 : 

/1/: initialization/OverloadingVarargs2. java 

7/ {CompileTimeError} (Won't compile) 


public class OverloadingVarargs2 { 
static void f(float i, Character... args) { 
System.out.printin("first"); 


$ 
static void f(Character... args) { 
System. out .print("second") ; 


} 
public static void main(String[] args) { 
tL, fa"); 
f(a", 'b"); 
g 
dh 


{CompileTimeError} 注 释 标 签 把 该 文件 排除 在 了 本 书 的 Ant 构 建 之 外 。 如 果 你 手动 编译 它 ， 
就 会 得 到 下 面 的 错误 消息 ; 


reference to f is ambiguous, both method fifloat,java.lang.Character...) 
in OverloadingVarargs2 and method f(java.lang.Character...) in 
Overloading Varargs2 match 


如 果 你 给 这 两 个 方法 都 添加 一 个 非 可 变 参数 ， 就 可 以 解决 问题 了 : 


/1/: initialization/OverloadingVarargs3. java 


public class OverloadingVarargs3 { 
static void f(float i, Character... args) { 
System.out.println("first") ; 


} 
static void f(char c, Character... args) { 
System.out..printin("second"); 


$. 
public static void main(String(] args) { 
f(1, 'a'); 
fat, "Dy; 
} 
} /* Output: 
first 
second 
Winx 


你 应 该 总 是 只 在 重 载 方法 的 一 个 版 本 上 使 用 可 变 参数 列表 ， 或 者 压根 就 不 是 用 它 。 

练习 19: (2) 写 一 个 类 ， 它 接受 一 个 可 变 参 数 的 String 数 组 。 验 证 你 可 以 向 该 方法 传递 一 个 
用 去 号 分 隔 的 String 列 表 ， 或 是 一 个 StringD 。 

练习 20: (1) 创建 一 个 使 用 可 变 参数 列表 而 不 是 普通 的 main0 语 法 的 main0。 打 印 所 产生 的 
| args 数 组 的 所 有 元 素 ， 并 用 各 种 不 同 数 量 的 命令 行 参数 来 测试 它 。 


| 5.9 枚 举 类 型 
| 在 Java SE5 中 添加 了 一 个 看 似 很 小 的 特性 ， 即 enum 关 键 字 ， 它 使 得 我 们 在 需要 群 组 并 使 用 
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枚 举 类 型 集 时 ， 可 以 很 方便 地 处 理 。 在 此 之 前 ， 你 需要 创建 一 个 整 型 常量 集 ， 但 是 这 些 枚 举 值 
并 不 会 必然 地 将 其 自身 的 取 值 限制 在 这 个 常量 集 的 范围 之 内 ， 因 此 它们 显得 更 有 风险 ， 且 更 难 
以 使 用 。 枚 举 类 型 属于 非常 普遍 的 需求 ，C、C++ 和 其 他 许多 语言 都 已 经 拥有 它 了 。 在 Java SES 
之 前 ，Java 程 序 员 在 需要 使 用 枚 举 类 型 时 ， 必 须 了 解 很 多 细节 并 需要 格外 仔细 ， 以 正确 地 产生 
enum 的 效果 。 现 在 Java 也 有 了 enum， 并 且 它 的 功能 比 C/C++ 中 的 枚 举 类 型 要 完备 得 多 。 下 面 是 
一 个 简单 的 例子 : 


/1/: initialization/Spiciness.java 
public enum Spiciness { 

NOT, MILD, MEDIUM, HOT, FLAMING 
} i~ 


这 里 创建 了 一 个 名 为 Spiciness 的 枚 举 类 型 ， 它 具有 5 个 具名 值 。 由 于 枚 举 类 型 的 实例 是 常 
i, 按照 命名 惯例 它们 都 用 大 写字 母 表示 (如 果 在 一 个 名 字 中 有 多 个 单词 ， 用 下 划 线 将 它 
们 隔 开 )。 

为 了 使 用 enum， 需 要 创建 一 个 该 类 型 的 引用 ， 并 将 其 赋值 给 某 个 实例 : 


//: initialization/SimpleEnumUse .java 
public class SimpleEnumUse { 
public static void main(String(] args) { 
Spiciness howHot = Spiciness. MEDIUM; 
System. out.printin(howHot) ; 





+ 
} /* Output: 
MEDIUM 
Wha 


在 你 创建 enum 时 ， 编 译 器 会 自动 添加 一 些 有 用 的 特性 。 例 如 ， 它 会 创建 toString0 方 法 ， 以 
便 你 可 以 很 方便 地 显示 某 个 enum 实 例 的 名 字 ， 这 正 是 上 面 的 打印 语句 如 何 产生 其 输出 的 答案 。 
编译 器 还 会 创建 ordinal0 方 法 ， 用 来 表示 某 个 特定 enum 常 量 的 声明 顺序 ， 以 及 static values() 方 
法 ， 用 来 按照 enum 常 量 的 声明 顺序 ， 产 生 由 这 些 常量 值 构成 的 数组 ; 


/1/; Anitialization/EnumOrder. java 
public class EnumOrder { 
public static void main(String{] args) { 
for(Spiciness s : Spiciness.values()) 
System.out.printin(s +", ordinal * + s.ordinal()); 


} 
} /* Output: 
NOT, ordinal © 
MILD, ordinal 1 
MEDIUM, ordinal 2 
HOT, ordinal 3 
FLAMING, ordinal 4 
Wh 


尽管 enum 看 起 来 像 是 一 种 新 的 数据 类 型 ， 但 是 这 个 关键 字 只 是 为 enum 生 成 对 应 的 类 时 ， 
产生 了 某 些 编译 器 行为 ， 因 此 在 很 大 程度 上 ， 你 可 以 将 enum 当 作 其 他 任何 类 来 处 理 。 事 实 上 ， 
enum 确 实 是 类 ， 并 且 具 有 自己 的 方法 。 

enum 有 一 个 特别 实用 的 特性 ， 即 它 可 以 在 switch 语 句 内 使 用 : 

//: initiatization/Burrito. java 


public class Burrito { 
Spiciness degree; 
public Burrito(Spiciness degree) { this.degree = degree; } 
public void describe() { 
System.out.print("This burrito is "); 
switch(degree) { 
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case NOT: — System.out.printin(*not spicy at all."); 
break; 

case MILD: 

case MEDIUM: System.out.printin(“a little hot."); 
break; 

case HOT: 

case FLAMING: 

default: System.out.printin(*maybe too hot."); 


} 
} 
public static void main(String[] args) { 
Burrito 
plain = new Burrito(Spiciness.NOT). 
BreenChile = new Burrito(Spiciness.MEDIUM), 
jalapeno = new Burrito(Spiciness.HOT) ; 
plain.describe(); 
greenChile.describe(); 
jalapeno. descr ibe(); 
} 
} /* Output: 
This burrito is not spicy at all. 
This burrito is a little hot. 
This burrito is maybe too hot. 
Wha 


由 于 switch 是 要 在 有 限 的 可 能 值 集合 中 进行 选择 ， 因 此 它 与 emum 正 是 绝 佳 的 组 合 。 请 注意 
enum 的 名 字 是 如 何 能 够 倍加 清楚 地 表明 程序 意欲 何 为 的 。 

大 体 上 ， 你 可 以 将 enum 用 作 另 外 一 种 创建 数据 类 型 的 方式 ， 然 后 直接 将 所 得 到 的 类 型 拿 来 
使 用 。 这 正 是 关键 所 在 ， 因 此 你 不 必 过 多 地 考虑 它们 。 在 Java SE5 引 进 enum 之 前 ， 你 必须 花费 
大 量 的 精力 去 保证 与 其 等 价 的 枚 举 类 型 是 安全 可 用 的 。 

这 些 介绍 对 于 你 理解 和 使 用 基本 的 enum 已 经 足够 了 ， 但 是 我 们 将 在 第 19 章 中 更 加 深入 地 探 
讨 它们 。 

练习 21: (1) 创建 一 个 enum， 它 包含 纸币 中 最 小 面值 的 6 种 类 型 。 通 过 values0 循 环 并 打印 每 
一 个 值 及 其 ordinal0。 

练习 22，(2) 在 前 面 的 例子 中 ， 为 enum 写 一 个 switch 语 句 ， 对 于 每 一 个 case， 输 出 该 特定 货 
币 的 描述 。 


5.10 总 结 


构造 器 ， 这 种 精巧 的 初始 化 机 制 ， 应 该 给 了 读者 很 强 的 暗示 : 初始 化 在 Java 中 占有 至 关 重 
要 的 地 位 。C++ 的 发 明 人 Bjame Stroustrup 在 设计 C++ 期 间 ， 在 针对 C 语 言 的 生产 效率 所 进行 的 最 
初 调查 中 发 现 ， 大 量 编程 错误 都 源 于 不 正确 的 初始 化 。 这 种 错误 很 难 发 现 ， 并 且 不 恰当 的 清理 
也 会 导致 类 似 问题 。 构 造 器 能 保证 正确 的 初始 化 和 清理 (没有 正确 的 构造 器 调用 ， 编 译 器 就 不 
允许 创建 对 象 )， 所 以 有 了 完全 的 控制 ， 也 很 安全 。  - 

在 C++ 中 ,“ 析 构 ” 相 当 重 要 ， 因 为 用 new 创 建 的 对 象 必须 明确 被 销毁 。 在 Java 中 ， 垃 圾 
回收 器 会 自动 为 对 象 释放 内 存 ， 所 以 在 很 多 场合 下 ， 类 似 的 清理 方法 在 Java 中 就 不 太 需 要 了 
(不 过 当 要 用 到 的 时 候 ， 你 就 只 能 自己 动手 了 )。 在 不 需要 类 似 析 构 函数 的 行为 的 时 候 ，Java 
的 垃圾 回收 器 可 以 极 大 地 简化 编程 工作 ， 而 且 在 处 理 内 存 的 时 候 也 更 安全 。 有 些 垃圾 回收 器 
甚至 能 清理 其 他 资源 ， 比 如 图 形 和 文件 句柄 。 然 而 ， 垃 圾 回收 器 确实 也 增加 了 运行 时 的 开销 。 
而 且 Java 解 释 器 从 来 就 很 慢 ， 所 以 这 种 开销 到 底 造成 了 多 大 的 影响 也 很 难看 出 。 随 着 时 间 的 
推移 ，Java 在 性 能 方面 已 经 取得 了 长 足 的 进步 ， 但 速度 问题 仍然 是 它 涉足 某 些 特定 编程 领域 
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的 障碍 。 

由 于 要 保证 所 有 对 象 都 被 创建 ， 构 造 器 实际 上 要 比 这 里 所 讨论 的 更 复杂 。 特 别 当 通过 组 
合 或 继承 生成 新 类 的 时 候 ， 这 种 保证 仍然 成 立 ， 并 且 需 要 一 些 附 加 的 语法 来 提供 支持 。 在 后 
面 的 章节 中 ， 读 者 将 学 习 到 有 关 组合 、 继 承 以 及 它们 对 构造 器 造成 的 影响 等 方面 的 知识 。 

所 选 习题 的 答案 都 可 以 在 名 为 The Thinking in Java Annotated Solution Guide 的 电子 文档 中 找 
到 ， 读 者 可 以 从 www.MindView.net 购 买 此 文档 。 





第 6 章 访问 权限 控制 


访问 控制 (或 隐藏 具体 实现 ) 与 “最 初 的 实现 并 不 恰当 ”有 关 。 

所 有 优秀 的 作者 ， 包 括 那些 编写 软件 的 程序 员 ， 都 清楚 其 著作 的 某 些 部 分 直至 重新 创作 的 
时 候 才 变 得 完美 ， 有 时 甚至 要 反复 重 写 多 次 。 如 果 你 把 一 个 代码 段 放 到 了 某 个 位 置 ， 等 过 一 会 
儿 回 头 再 看 时 ， 有 可 能 会 发 现 有 更 好 的 方式 去 实现 相同 的 功能 。 这 正 是 重 构 的 原动力 之 一 ， 重 
构 即 重 写 代码 ， 以 使 得 它 更 可 读 、 更 易 理解 ， 并 因此 而 更 具 可 维护 性 。 

但 是 ， 在 这 种 修改 和 完善 代码 的 愿望 之 下 ， 也 存在 着 巨大 的 压力 。 通 常 总 是 会 有 一 些 消费 
者 (客户 端 程序 员 ) 需要 你 的 代码 在 某 些 方面 保持 不 变 。 因 此 你 想 改变 代码 ， 而 他 们 却 想 让 代 
码 保持 不 变 。 由 此 而 产生 了 在 面向 对 象 设计 中 需要 考虑 的 一 个 基本 问题 :“ 如 何 把 变动 的 事物 与 
保持 不 变 的 事物 区 分 开 来 ”。 

这 对 类 库 (library) 而 言 尤为 重要 。 该 类 库 的 消费 者 必须 依赖 他 所 使 用 的 那 部 分 类 库 ， 并 且 
能 够 知道 如 果 类 库 出 现 了 新 版 本 ， 他 们 并 不 需要 改写 代码 。 从 另 一 个 方面 来 说 ， 类 库 的 开发 者 
必须 有 权限 进行 修改 和 改进 ， 并 确保 客户 代码 不 会 因为 这 些 改动 而 受到 影响 。 

这 一 目标 可 以 通过 约定 来 达到 。 例 如 ， 类 库 开发 者 必须 同意 在 改动 类 库 中 的 类 时 不 得 删除 
任何 现 有 方法 ， 因 为 那样 会 破坏 客户 端 程序 员 的 代码 。 但 是 ， 与 之 相反 的 情况 会 更 加 同 手 。 在 
有 域 〈 即 数据 成 员 ) 存在 的 情况 下 ， 类 库 开发 者 要 怎样 才能 知道 究竟 都 有 哪些 域 已 经 被 客户 端 
程序 员 所 调用 了 呢 ? 这 对 于 方法 仅 为 类 的 实现 的 一 部 分 ， 因 此 并 不 想 让 客户 端 程序 员 直 接 使 用 
的 情况 来 说 同样 如 此 。 如 果 程 序 开发 者 想 要 移 除 旧 的 实现 而 要 添加 新 的 实现 时 ， 结 果 将 会 怎样 
Ye? 改动 任何 一 个 成 员 都 有 可 能 破坏 客户 端 程序 员 的 代码 。 于 是 类 库 开发 者 会 手脚 被 缚 ， 无 法 
对 任何 事物 进行 改动 。 

为 了 解决 这 一 问题 ，Java 提 供 了 访问 权限 修饰 词 ， 以 供 类 库 开 发 人 员 向 客户 端 程序 员 指明 
哪些 是 可 用 的 , 哪些 是 不 可 用 的 。 访 问 权限 控制 的 等 级 ， 从 最 大 权限 到 最 小 权限 依次 为 : public, 
protected、 包 访问 权限 (没有 关键 词 ) 和 private。 根 据 前 述 内 容 ， 读 者 可 能 会 认为 ， 作 为 一 名 
类 库 设 计 员 ， 你 会 尽 可 能 将 一 切 方法 都 定 为 private， 而 仅 向 客户 端 程序 员 公 开 你 愿意 让 他 们 使 
用 的 方法 。 这 样 做 是 完全 正确 的 ， 尽 管 对 于 那些 经 常 使 用 别 的 语言 (特别 是 C 语 言 ) 编写 程序 并 
在 访问 事物 时 不 受 任何 限制 的 人 而 言 ， 这 与 他 们 的 直觉 相 违背 。 到 了 本 章 末 ， 读 者 将 会 信服 
Java 的 访问 权限 控制 的 价值 。 

不 过 ， 构 件 类 库 的 概念 以 及 对 于 谁 有 权 取 用 该 类 库 构 件 的 控制 问题 都 还 是 不 完善 的 。 其 中 
仍旧 存在 着 如 何 将 构件 捆绑 到 一 个 内 聚 的 类 库 单元 中 的 问题 。 对 于 这 一 点 ，Java 用 关键 字 
Package 加 以 控制 ， 而 访问 权限 修饰 词 会 因 类 是 存在 于 一 个 相同 的 包 ， 还 是 存在 于 一 个 单独 的 包 
而 受到 影响 。 为 此 ， 要 开始 学 习 本 章 ， 首 先 要 学 习 如 何 将 类 库 构件 置 于 包 中 ， 然 后 就 会 理解 访 
问 权限 修饰 词 的 全 部 含义 。 


© 请 查看 Refactoring: Improving the Design of Existing Code， 作 者 Martin Fowler 等 (Addison-Wesley,1999)。 侦 尔 会 
有 某 些 人 对 重 构 抱 有 异议 ， 他 们 认为 这 些 代码 工作 得 极 好 ， 重 构 它们 无 异 于 浪费 时 间 。 这 种 思维 方式 的 问题 
在 于 对 于 项 目 所 需 的 时 间 和 资金 来 说 ， 最 大 的 部 分 并 非 投入 到 了 最 初 的 代码 编写 上 ， 而 是 投入 到 了 代码 的 维 
护 上 。 因 此 ， 使 代码 更 加 易于 理解 就 意味 着 节省 了 大 量 的 金钱 。 
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61 @: 库 单元 


包 内 包含 有 一 组 类 ， 它 们 在 单一 的 名 字 空 间 之 下 被 组 织 在 了 一 起 。 
例如 ， 在 Java 的 标准 发 布 中 有 一 个 工具 库 ， 它 被 组 织 在 javautil 名 字 空 间 之 下 。java.util 中 
有 一 个 叫做 ArrayList 的 类 ， 使 用 ArrayList 的 一 种 方式 是 用 其 全 名 javautil.ArrayList 来 指定 。 


/1/: access/FullQualification. java 


public class FullQualification { 
public static void main(String[] args) { 
java.util.ArrayList list = new java.util.ArrayList(); 
} 
} Mi~ 


这 立刻 就 使 程序 变 得 很 元 长 了 ， 因 此 你 可 能 想 转 而 使 用 import 关 键 字 。 如 果 你 想 要 导入 单 
个 的 类 ， 可 以 在 import 语 句 中 命名 该 类 : 


//: access/SingleImport. java 
import java.util.Arraylist; 


public class SingleImport { 
public static void main(String] args) { 
ArrayList list = new java.util.Arraylist(); 


) 

Mh 

现在 ， 就 可 以 不 用 限定 地 使 用 ArrayList 了 。 但 是 ， 这 样 做 java.util 中 的 其 他 类 仍旧 是 都 不 可 
用 的 。 要 想 导 入 其 中 所 有 的 类 ， 只 需要 使 用 “*”"， 就 像 在 本 书 剩余 部 分 的 示例 中 所 看 到 的 那样 

import java.util.*; 

我 们 之 所 以 要 导入 ， 就 是 要 提供 一 个 管理 名 字 空间 的 机 制 。 所 有 类 成 员 的 名 称 都 是 彼此 隔 - 
离 的 。A 类 中 的 方法 f0 与 B 类 中 具有 相同 特征 标记 (参数 列表 ) 的 方法 f0 不 会 彼此 冲突 。 但 是 如 
果 类 名 称 相互 冲突 又 该 怎么 办 呢 ? 假设 你 编写 了 一 个 Stack 类 并 安装 到 了 一 台 机 器 上 ， 而 该 机 器 
上 已 经 有 了 一 个 别人 编写 的 Stack 类 ， 我 们 该 如 何 解决 呢 ? 由 于 名 字 之 间 的 潜在 冲突 ， 在 Java 中 
对 名 称 空间 进行 完全 控制 并 为 每 个 类 创建 唯一 的 标识 符 组 合 就 成 为 了 非常 重要 的 事情 。 

到 目前 为 止 ， 书 中 大 多 数 示例 都 存 于 单一 文件 之 中 ， 并 专 为 本 地 使 用 (local use) 而 设计 ， 
因而 尚未 受到 包 名 的 干扰 。 这 些 示例 实际 上 已 经 位 于 包 中 了 : 即 示 命名 包 ， 或 称 为 默认 包 。 这 
当然 也 是 一 种 选择 ， 而 且 为 了 简单 起 见 ， 在 本 书 其 他 部 分 都 尽 可 能 地 使 用 了 此 方法 。 不 过 如 果 
你 正在 准备 编写 对 在 同一 台 机 器 上 共存 的 其 他 Java 程 序 友好 的 类 库 或 程序 的 话 ， 就 需要 考虑 如 
何 防止 类 名 称 之 间 的 冲突 问题 。 

当 编写 一 个 Java 源 代码 文件 时 ， 此 文件 通常 被 称 为 编译 单元 (有 时 也 被 称 为 转译 单元 ) 。 每 
个 编译 单元 都 必须 有 一 个 后 组 名 ,java， 而 在 编译 单元 内 则 可 以 有 一 个 public 类 ， 该 类 的 名 称 必 
须 与 文件 的 名 称 相同 (包括 大 小 写 ， 但 不 包括 文件 的 后 组 名 .java) 。 每 个 编译 单元 只 能 有 一 个 
public 类 ， 否 则 编译 器 就 不 会 接受 。 如 果 在 该 编译 单元 之 中 还 有 额外 的 类 的 话 ， 那 么 在 包 之 外 
的 世界 是 无 法 看 见 这些 类 的 ， 这 是 因为 它们 不 是 public 类 ， 而 且 它 们 主要 用 来 为 主 public 类 提供 
支持 。 

6.1.1 代码 组 织 

当 编 译 一 个 java 文 件 时， 在 java 文 件 中 的 每 个 类 都 会 有 一 个 输出 文件 ， 而 该 输出 文件 的 名 
称 与 java 文 件 中 每 个 类 的 名 称 相同 ， 只 是 多 了 一 个 后 级 名 .class。 因 此 ， 在 编译 少量 ,java 文件 之 
后 ,会 得 到 大 量 的 .class 文 件 。 如 果 用 编译 型 语言 编写 过 程序 ， 那 么 对 于 编译 器 产生 一 个 中 间 文 
件 (通常 是 一 个 obj 文 件 )， 然 后 再 与 通过 链接 器 (用 以 创建 一 个 可 执行 文件 ) 或 类 库 产生 器 
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(librarian， 用 以 创建 一 个 类 库 ) 产生 的 其 他 同类 文件 捆绑 在 一 起 的 情况 ， 可 能 早已 司空 见 惯 。 
但 这 并 不 是 Java 的 工作 方式 。Java 可 运行 程序 是 一 组 可 以 打包 并 压缩 为 一 个 Java 文档 文件 (JAR， 
使 用 Java 的 jar 文 档 生成 器 ) 的 .class 文 件 。Java 解 释 器 负责 这 些 文件 的 查找 、 装 载 和 解释 ? 。 

类 库 实际 上 是 一 组 类 文件 。 其 中 每 个 文件 都 有 一 个 public 类 ， 以 及 任意 数量 的 非 public 类 。 
因此 每 个 文件 都 有 一 个 构件 。 如 果 希 望 这 些 构件 〈 每 一 个 都 有 它们 自己 的 独立 的 ,java 和 ,class 文 
件 ) 从 属于 同一 个 群 组 ， 就 可 以 使 用 关键 字 package。 

如 果 使 用 package 语 句 ， 它 必须 是 文件 中 除 注释 以 外 的 第 一 句 程序 代码 。 在 文件 起 始 处 写 : 

package access; 

就 表示 你 在 声明 该 编译 单元 是 名 为 access 的 类 库 的 一 部 分 。 或 者 换 种 说 法 ， 你 正在 声明 该 编译 单 
元 中 的 public 类 名 称 是 位 于 access 名 称 的 保护 们 下 。 任 何 想 要 使 用 该 名 称 的 人 都 必须 使 用 前 面 给 
出 的 选择 ， 指 定金 名 或 者 与 access 结 合 使 用 关键 字 import。( 请 注意 ，Java 包 的 命名 规则 全 部 使 
用 小 写字 母 ， 包 括 中 间 的 字 也 是 如 此 。) 

例如 ， 假 设 文件 的 名 称 是 MyClass.java， 这 就 意味 着 在 该 文件 中 有 且 只 有 一 个 public 类 ， 该 
类 的 名 称 必须 是 MyClass (注意 大 小 写 ) : 


//: access/mypackage/MyClass. java 
package access.mypackage; 


public class MyClass { 
V4 ven 
} Mi~ 
现在 ， 如 果 有 人 想 用 MyClass 或 者 是 access 中 的 任何 其 他 public 类 ， 就 必须 使 用 关键 字 
import 来 使 access 中 的 名 称 可 用 。 另 一 个 选择 是 给 出 完整 的 名 称 : 


//: access/QualifiedMyClass. java 


public class QualifiedMyClass { 
public static void main(String[] args) { 
access.mypackage.MyClass m = 
new access.mypackage .MyClass() ; 


) hr- 
关键 字 import 可 使 之 更 加 简洁 


//: access/ImportedMyClass. java 
import access.mypackage.*; 


public class ImportedMyClass { 
public static void main(String{] args) { 
MyClass m = new MyClass(); 


} 

} M~ 

身 为 一 名 类 库 设计 员 ， 很 有 必要 牢记 : package 和 import 关 键 字 允许 你 做 的 ， 是 将 单一 的 
全 局 名 字 空间 分 割 开 ， 使 得 无 论 多 少 人 使 用 Internet 以 及 Java 开 始 编写 类 ， 都 不 会 出 现 名 称 冲突 
问题 。 
6.1.2 创建 独一无二 的 包 名 

读者 也 许 会 发 现 ， 既 然 一 个 包 从 未 真正 将 被 打包 的 东西 包装 成 单一 的 文件 ， 并 且 一 个 包 可 
以 由 许多 .class 文件 构成 ， 那 么 情况 就 有 点 复杂 了 。 为 了 避免 这 种 情况 的 发 生 ， 一 种 合乎 逻辑 的 
做 法 就 是 将 特定 包 的 所 有 .class 文件 都 置 于 一 个 目录 下 。 也 就 是 说 ， 利 用 操作 系统 的 层次 化 的 文 


O Java 中 并 不 强求 必须 要 使 用 解释 器 。 因 为 存在 用 来 生成 一 个 单一 的 可 执行 文件 的 本 地 代码 Java 编 译 器 。 
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件 结构 来 解决 这 一 问题 。 这 是 Java 解 决 混乱 问题 的 一 种 方式 ， 读 者 还 会 在 我 们 介绍 jar 工 具 的 时 
候 看 到 另 一 种 方式 。 

将 所 有 的 文件 收入 一 个 子 目录 还 可 以 解决 另外 两 个 问题 : 怎样 创建 独一无二 的 名 称 以 及 怎 
样 查找 有 可 能 隐藏 于 目录 结构 中 某 处 的 类 。 这 些 任务 是 通过 将 -class 文件 所 在 的 路 径 位 置 编码 成 
Package 的 名 称 来 实现 的 。 按 照 惯 例 ，package 名 称 的 第 一 部 分 是 类 的 创建 者 的 反 顺序 的 Internet 
域名 。 如果 你 遵照 惯例 ，Internet 域 名 应 是 独一无二 的 ， 因 此 你 的 package 名 称 也 将 是 独一无二 的 ， 
也 就 不 会 出 现 名 称 冲 突 的 问题 了 (也 就 是 说 ， 只 有 在 你 将 自己 的 域名 给 了 别人 ， 而 他 又 以 你 曾 
经 使 用 过 的 路 径 名 称 来 编写 Java 程 序 代码 时 ， 才 会 出 现 冲突 )。 当 然 ， 如 果 你 没有 自己 的 域名 ， 
你 就 得 构造 一 组 不 大 可 能 与 他 人 重复 的 组 合 ( 例 如 你 的 姓名 ) ， 来 创立 独一无二 的 package 名 称 。 
如 果 你 打算 发 布 你 的 Java 程 序 代码 ， 稍 微 花 点 力气 去 取得 一 个 域名 ， 还 是 很 有 必要 的 。 

此 技巧 的 第 二 部 分 是 把 package 名 称 分 解 为 你 机 器 上 的 一 个 目录 。 所 以 当 Java 程 序 运行 并 且 
需要 加 载 .class 文 件 的 时 候 ， 它 就 可 以 确定 .class 文 件 在 目录 上 所 处 的 位 置 。 

Java 解 释 器 的 运行 过 程 如 下 : 首先 ， 找 出 环境 变量 CLASSPATHe (可 以 通过 操作 系统 来 设 
置 ， 有 时 也 可 通过 安装 程序 -用 来 在 你 的 机 器 上 安装 Java 或 基于 Java 的 工具 -来 设置 ) 。 
CLASSPATH 人 包含 一 个 或 多 个 目录 ， 用 作 查 找 .class 文 件 的 根 目录 。 从 根 目录 开始 ， 解 释 器 获取 包 
的 名 称 并 将 每 个 句点 替换 成 反 斜 本 ， 以 从 CLASSPATH 根 中 产生 一 个 路 径 名 称 (FUE, package 
foo.bar-baz 就 变 成 为 foo\bar\baz 或 foo/bar/baz 或 其 他 ， 这 一 切取 决 于 操作 系统 )。 得 到 的 路 径 会 
与 CLASSPATH 中 的 各 个 不 同 的 项 相连 接 ， 解 释 器 就 在 这 些 目录 中 查找 与 你 所 要 创建 的 类 名 称 相 
关 的 .class 文件 。( 解 释 器 还 会 去 查找 某 些 涉及 Java 解 释 器 所 在 位 置 的 标准 目录 。) 

为 了 理解 这 一 点 ， 以 我 的 域名 MindView.net 为 例 。 把 它 的 顺序 倒 过 来 ， 并 且 将 其 全 部 转换 
为 小 写 ，netmindview 就 成 了 我 所 创建 的 类 的 独一无二 的 全 局 名 称 。(com、edu、org 等 扩展 名 先 
前 在 Java 包 中 都 是 大 写 的 ， 但 在 Java2 中 一 切 都 已 改观 ， 包 的 整个 名 称 全 都 变 成 了 小 写 。) 若 我 决 
定 再 创建 一 个 名 为 simple 的 类 库 ， 我 可 以 将 该 名 称 进 一 步 细 分 ， 于 是 我 可 以 得 到 一 个 包 的 名 称 
如 下 ， 

package net.mindview. simple; 

现在 ， 这 个 包 名 称 就 可 以 用 作 下 面 两 个 文件 的 名 字 空 间 保护 们 了 ; 


//: net/mindview/simple/Vector. java 
// Creating a package. 
package net.mindview.simple; 


public class Vector { 
public Vector() { 
System. out. printin(“net.mindview. simple. Vector"); 


} Mi~ 
如 前 所 述 ，package 语 句 必须 是 文件 中 的 第 一 行 非 注 释 程 序 代 码 。 第 二 个 文件 看 起 来 也 极其 
相似 : 

//: net/mindview/simple/List. java 
// Creating a package. 
package net .mindview. simple; 
public class List { 

public List() { 

System. out.printIn("net.mindview.simple.List"); 

} 

} Mi~ 


O 当 提 及 环境 变量 时 ， 将 用 到 大 写字 母 (CLASSPATH)。 
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这 两 个 文件 均 被 置 于 我 的 系统 的 子 目录 下 : 

C:\DOC\JavaT\net\mindview\simple 

(注意 ， 在 本 书 的 每 一 个 文件 中 的 第 一 行 注释 都 指定 了 该 文件 在 源 代码 目录 树 中 的 位 置 ， 这 
个 信息 将 由 针对 本 书 的 自动 代码 抽取 工具 使 用 。) 

如 果 沿 此 路 径 往 回 看 ， 可 以 看 到 包 的 名 称 com.bruceeekelLsimple， 但 此 路 径 的 第 一 部 分 怎样 
办 呢 ? 它 将 由 环境 变量 CLASSPATH 关 照 ， 在 我 的 机 器 上 是 ， 

CLASSPATH=. ;D:\JAVA\LIB;C:\DOC\JavaT 

可 以 看 到 ，CLASSPATH 可 以 包含 多 个 可 供 选择 的 查询 路 径 。 

但 在 使 用 JAR 文 件 时 会 有 一 点 变化 。 必 须 在 类 路 径 中 将 JAR 文 件 的 实际 名 称 写 清楚 ， 而 不 仅 
是 指明 它 所 在 位 置 的 目录 。 因 此 ， 对 于 一 个 名 为 grapejar 的 JAR 文 件 ， 类 路 径 应 包括 ， 

CLASSPATH=. ;D:\JAVA\LIB;C:\flavors\grape. jar 
一 旦 类 路 径 得 以 正确 建立 ， 下 面 的 文件 就 可 以 放 于 任何 目录 之 下 : 


//: access/LibTest.java 
// Uses the library. 
‘import net.mindview. simple. *; 














public class LibTest { 
public static void main(String] args) { 
Vector v = new Vector(); 
List 1 = new List(); 
} 
} /* Output: 
net.mindview. simple. Vector 
net.mindview.simple.List 
I~ 


当 编 译 器 碰 到 simple 库 的 import 语 句 时 ， 就 开始 在 CLASSPATH 所 指定 的 目录 中 查找 ， 查 找 
子 目 录 netvmindview\simple， 然 后 从 已 编译 的 文件 中 找 出 名 称 相符 者 (对 Vector 而 言 是 
Vector.class， 对 List 而 言 是 List.class) 。 请 注意 ，Vector 和 List 中 的 类 以 及 要 使 用 的 方法 都 必须 是 
public 的 。 

对 于 使 用 Java 的 新 手 而 言 ， 设 立 CLASSPATH 是 很 麻烦 的 一 件 事 (我 最 初 使 用 时 就 是 这 样 
的 ) ， 为 此 ，Sun 将 Java2 中 的 JDK 改 造 得 更 聪明 了 一 些 。 在 安装 后 你 会 发 现 ， 即 使 你 未 设立 
CLASSPATH， 你 也 可 以 编译 并 运行 基本 的 Java 程 序 。 然 而 ， 要 编译 和 运行 本 书 的 源码 包 (从 
www.MindView.net 网 站 可 以 取得 ) ， 就 得 向 你 的 CLASSPATH 中 添加 本 书 程序 代码 树 中 的 基 目 
录 了 。 

练习 1: (1) 在 某 个 包 中 创建 一 个 类 ， 在 这 个 类 所 处 的 包 的 外 部 创建 该 类 的 一 个 实例 。 

冲突 

如 果 将 两 个 含有 相同 名 称 的 类 库 以 “*” 形 式 同时 导入 ， 将 会 出 现 什么 情况 呢 ? 例如 ， 假 设 
某 程 序 这 样 写 : 


import net.mindview.simple.*; 
import java.util.*; 


由 于 javautil.* 也 含有 一 个 Vector 类 ， 这 就 存在 潜在 的 冲突 。 但 是 只 要 你 不 写 那些 导致 冲突 
的 程序 代码 ， 就 不 会 有 什么 问题 -这 样 很 好 ， 否 则 就 得 做 很 多 的 类 型 检查 工作 来 防止 那些 根本 不 
会 出 现 的 冲突 。 
如 果 现 在 要 创建 一 个 Vector 类 的 话 ， 就 会 产生 冲突 : 
1 Vector v = new Vector(); 


这 行 到 底 取 用 的 是 哪个 Vector 类 ? 编译 器 不 知道 ， 读 者 同样 也 不 知道 。 于 是 编译 器 提出 错 
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误 信 息 ， 强 制 你 明确 指明 。 举 例 说 明 ， 如 果 想 要 一 个 标准 的 Java Vector 类 ， 就 得 这 样 写 : 

java.util.Vector v = new java.util.Vector(); 

由 于 这 样 可 以 完全 指明 该 Vector 类 的 位 置 (配合 CLASSPATH) ， 所 以 除非 还 要 使 用 java.util 
中 的 其 他 东西 ， 否 则 就 没有 必要 写 import javautil.* 语 句 了 。 

或 者 ， 可 以 使 用 单个 类 导入 的 形式 来 防止 冲突 ， 只 要 你 在 同一 个 程序 中 没有 使 用 有 冲突 的 
名 字 (在 使 用 了 有 冲突 名 字 的 情况 下 ， 必 须 返 回 到 指定 全 名 的 方式 ) 。 

练习 2: (1) 将 本 节 中 的 代码 片段 改写 为 完整 的 程序 ， 并 校 验 实际 所 发 生 的 冲突 。 


6.1.3 定制 工具 库 

具备 了 这 些 知识 以 后 ， 现 在 就 可 以 创建 自己 的 工具 库 来 减少 或 消除 重复 的 程序 代码 了 。 例 
如 ， 我 们 已 经 用 到 的 System.out.printin0 的 别名 可 以 减少 输入 负担 ， 这 种 机 制 可 以 用 于 名 为 Print 
的 类 中 ， 这 样 ， 我 们 在 使 用 该 类 时 可 以 用 一 个 更 具 可 读 性 的 静态 import 语 名 来 导入 : 


//: net/mindview/util/Print. java 

// Print methods that can be used without 

// qualifiers, using Java SES static imports: 
package net.mindview.util; 

import java.to.*; 


public class Print { 
// Print with a newline: 
public static void print(Object obj) { 
System.out.printin(obj) ; 


} 

// Print a newline by itself: 

public static void print() { 
System.out.printin(); 


// Print with no line break: 
public static void printnb(Object obj) { 
System. out. print (obj); 


} 

// The new Java SES printf() (from C): 

public static PrintStream 

printf (String format, Object... args) { 
return System.out.printf (format, args); 


} 
} Ii~ 
可 以 使 用 打印 便捷 工具 来 打印 String ， 无 论 是 需要 换行 (print()) 还 是 不 需要 换行 
(printnbO) 。 


可 以 猜 到 ， 这 个 文件 的 位 置 一 定 是 在 某 个 以 一 个 CLASSPATH 位 置 开 始 ， 然 后 接着 是 
net/mindview 的 目录 下 。 编 译 完 之 后 ， 就 可 以 用 import statie 语 句 在 你 的 系统 上 使 用 静态 的 
print0 和 printnb0 方 法 了 。 


//: access/PrintTest.java 
// Uses the static printing methods in Print. java. 
import static net.mindview.util.Print.*; 


public class PrintTest { 
public static void main(String[] args) { 
print(*Available from now on!*); 
print (100); 
print (1@0L); 
print (3.14159) ; 


3 
} /* Output: 
Available from now on! 
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100 
3.14159 
#77:~ 


这 个 类 库 的 第 二 个 构件 可 以 是 在 第 4 章 中 引入 的 range0 方 法 ， 它 使 得 foreach 语 法 可 以 用 于 简 
单 的 整数 序列 ; 


//: net/mindview/util/Range. java 
71 Array creation methods that can be used without 
// qualifiers, using Java SES static imports: 
package net.mindview.util; 


public class Range { 
// Produce a sequence [@..n) 
public static int[] range(int n) { 
int[] result = new int[n]; 
for(int 1 = 
result[1] 
return result; 


< ni itt) 





// Produce a sequence [start..end) 
public static int[] range(int start, int end) { 
int sz = end - start; 
int[] result = new int[sz]; 
for(int 1 = @; 1 < sz; i++) 
result(i] = start + i; 
return result; 


// Produce a sequence (start..end) incrementing by step 
public static int{] range(int start, int end, int step) { 
int sz = (end - start)/step; 
int[] result = new int[sz]; 
for(int i = 9; 1 < sz; i++) 
result(i] = start + (i * step); 
return result; 


} 
} Mi~ 


从 现在 开始 ， 你 无 论 何 时 创建 了 有 用 的 新 工具 ， 都 可 以 将 其 添加 到 你 自己 的 类 库 中 。 你 将 
看 到 在 本 书 中 还 有 更 多 的 构件 添加 到 了 net.mindview.util 类 库 中 。 
6.1.4 用 import 改 变 行为 

Java 没 有 C 的 条 件 编译 功能 ， 该 功能 可 以 使 你 不 必 更 改 任何 程序 代码 ， 就 能 够 切换 开关 并 产 
生 不 同 的 行为 。Java 去 掉 此 功能 的 原因 可 能 是 因为 C 在 绝 大 多 数 情况 下 是 用 此 功能 来 解决 跨 平台 
问题 的 ， 即 程序 代码 的 不 同 部 分 是 根据 不 同 的 平台 来 编译 的 。 由 于 Java 自 身 可 以 自动 跨越 不 同 
的 平台 ， 因 此 这 个 功能 对 Java 而 言 是 没有 必要 的 。 

然而 ， 条 件 编译 还 有 其 他 一 些 有 价值 的 用 途 。 调 试 就 是 一 个 很 常见 的 用 途 。 调 试 功能 在 开 
发 过 程 中 是 开启 的 ， 而 在 发 布 的 产品 中 是 禁用 的 。 可 以 通过 修改 被 导入 的 package 的 方法 来 实现 
这 一 目的 ， 修 改 的 方法 是 将 你 程序 中 用 到 的 代码 从 调试 版 改 为 发 布 版 。 这 一 技术 可 以 适用 于 任 
何 种 类 的 条 件 代码 。 

练习 3: (2) 创建 两 个 包 : debug 和 debugoff， 它 们 都 包含 一 个 相同 的 类 ， 该 类 有 一 个 debug0 
方法 。 第 一 个 版 本 显示 发 送 给 控制 台 的 String 参 数 ， 而 第 二 个 版 本 什么 也 不 做 。 使 用 静态 import 
语句 将 该 类 导入 到 一 个 测试 程序 中 ， 并 示范 条 件 编译 效果 。 
6.1.5 对 使 用 包 的 忠告 

务必 记 住 ， 无 论 何 时 创建 包 ， 都 已 经 在 给 定 包 的 名 称 的 时 候 隐 含 地 指定 了 目录 结构 。 这 个 
包 必 须 位 于 其 名 称 所 指定 的 目录 之 中 ， 而 该 目录 必须 是 在 以 CLASSPATH 开 始 的 目录 中 可 以 查询 
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到 的 。 最 初 使 用 关键 字 package， 可 能 会 有 一 点 不 顺 ， 因 为 除非 遵守 “ 包 的 名 称 对 应 目录 路 径 ” 
的 规则 ， 否 则 将 会 收 到 许多 出 乎 意料 的 运行 时 信息 ， 告 知 无 法 找到 特定 的 类 ， 哪 怕 是 这 个 类 就 
位 于 同一 个 目录 之 中 。 如 果 你 收 到 类 似 信息 ， 就 用 注释 掉 package 语 句 的 方法 试 一 下 ， 如 果 这 样 
程序 就 能 运行 的 话 ， 你 就 可 以 知道 问题 出 在 哪里 了 。 

注意 ， 编 译 过 的 代码 通常 放置 在 与 源 代码 的 不 同 目录 中 ， 但 是 必须 保证 JVN 使 用 CLASSPATH 
可 以 找到 该 路 径 。 


6.2 Java 访 问 权 限 修饰 词 


public、protected 和 private 这 几 个 Java 访 问 权限 修饰 词 在 使 用 时 ， 是 置 于 类 中 每 个 成 员 的 
定义 之 前 的 -无 论 它 是 一 个 域 还 是 一 个 方法 。 每 个 访问 权限 修饰 词 仅 控制 它 所 修饰 的 特定 定义 的 
访问 权 。 

如 果 不 提供 任何 访问 权限 修饰 词 ， 则 意味 着 它 是 “ 包 访问 权限 ”"。 因 此 ， 无 论 如 何 ， 所 有 事 
物 都 具有 某 种 形式 的 访问 权限 控制 。 在 以 下 几 节 中 ， 读 者 将 学 习 各 种 类 型 的 访问 权限 。 

6.2.1 包 访 问 权限 

本 章 之 前 的 所 有 示例 都 没有 使 用 任何 访问 权限 修饰 词 。 默 认 访问 权限 没有 任何 关键 字 ， 但 
通常 是 指 包 访问 权限 〈 有 时 也 表示 成 为 friendly)。 这 就 意味 着 当前 的 包 中 的 所 有 其 他 类 对 那个 成 
员 都 有 访问 权限 ， 但 对 于 这 个 包 之 外 的 所 有 类 ， 这 个 成 员 却 是 private。 由 于 一 个 编译 单元 ( 即 
一 个 文件 )， 只 能 隶属 于 一 个 包 ， 所 以 经 由 包 访问 权限 ， 处 于 同一 个 编译 单元 中 的 所 有 类 彼此 之 
间 都 是 自动 可 访问 的 。 

包 访问 权限 允许 将 包 内 所 有 相关 的 类 组 合 起 来 ， 以 使 它们 彼此 之 间 可 以 轻松 地 相互 作用 。 
当 把 类 组 织 起 来 放 进 一 个 包 内 之 时 ， 也 就 给 它们 的 包 访问 权限 的 成 员 赋 予 了 相互 访问 的 权限 ， 
你 “拥有 ”了 该 包 内 的 程序 代码 。“ 只 有 你 拥有 的 程序 代码 才 可 以 访问 你 所 拥有 的 其 他 程序 代码 ”， 
这 是 合理 的 。 应 该 说 ， 包 访问 权限 为 把 类 群 聚 在 一 个 包 中 的 做 法 提供 了 意义 和 理由 。 在 许多 语 
言 中 ， 在 文件 内 组 织 定义 的 方式 是 任意 的 ， 但 在 Java 中 ， 则 要 强制 你 以 一 种 合理 的 方式 对 它们 
加 以 组 织 。 另 外 ， 你 可 能 还 想 要 排除 这 样 的 类 一 它们 不 应 该 访问 在 当前 包 中 所 定义 的 类 。 

类 控制 着 哪些 代码 有 权 访问 自己 的 成 员 。 其 他 包 内 的 类 不 能 刚 一 上 来 就 说 ;“ 史 ， 我 是 Bob 
的 朋友 。” 并 且 还 想 看 到 Bob 的 protected、 包 访问 权限 和 private 成 员 。 取 得 对 某 成 员 的 访问 权 的 
唯一 途径 是 : 

1. 使 该 成 员 成 为 public。 于 是 ， 无 论 是 谁 ， 无 论 在 哪里 ， 都 可 以 访问 该 成 员 。 

2. 通过 不 加 访问 权限 修饰 词 并 将 其 他 类 放置 于 同一 个 包 内 的 方式 给 成 员 赋予 包 访问 权 。 于 
是 包 内 的 其 他 类 也 就 可 以 访问 该 成 员 了 。 

3. 在 第 7 章 将 会 介绍 继承 技术 ， 届 时 读者 将 会 看 到 继承 而 来 的 类 既 可 以 访问 public 成 员 也 可 
以 访问 protected 成 员 〈 但 访问 private 成 员 却 不 行 )。 只 有 在 两 个 类 都 处 于 同一 个 包 内 时 ， 它 才 可 
以 访问 包 访问 权限 的 成 员 。 但 现在 不 必 担 心 继承 和 protected。 

4. 提供 访问 器 (accessor) 和 变异 器 (mutator) 方法 (也 称 作 get/set 方 法 )， 以 读 取 和 改变 数 
值 。 正 如 将 在 第 22 章 中 看 到 的 ， 对 OOP 而 言 ， 这 是 最 优雅 的 方式 ， 而 且 这 也 是 JavaBeans 的 基本 
原理 。 

6.2.2 public: 接口 访问 权限 

使 用 关键 字 public， 就 意味 着 public 之 后 紧 跟着 的 成 员 声明 自己 对 每 个 人 都 是 可 用 的 ， 尤 其 

是 使 用 类 库 的 客户 程序 员 更 是 如 此 。 假 设 定义 了 一 个 包含 下 面 编译 单元 的 dessert 包 : 
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//: access/dessert/Cookie. java 
// Creates a library. 
package access.dessert; 


public class Cookie { 
public Cookie() { 
System.out.printin("Cookie constructor"); 
} 
void bite() { System.out.printin("bite"); } 
} :~ 


记 住 ，Cookie.java 文 件 必须 位 于 名 为 dessert 的 子 目 录 之 中 ， 该 子 目录 在 access ( 意 指 本 书 第 
6 章 ) 下 ， 而 c05 则 必须 位 于 CLASSPATH 指 定 的 众多 路 径 的 其 中 之 一 的 下 边 。 不 要 错误 地 认为 
Java 总 是 将 当前 目录 视 作 是 查找 行为 的 起 点 之 一 。 如 果 你 的 CLASSPATH 之 中 缺少 一 个 “.” 作 为 
路 径 之 一 的 话 ，Java 就 不 会 查找 那里 。 

现在 如 果 创 建 了 一 个 使 用 Cookie 的 程序 : 


//: access/Dinner.java 
// Uses the library. 
import access.dessert.*; 


public class Dinner { 
public static void main(String(] args) { 
Cookie x = new Cookie(); 
/ME x.bite(); // Can't access 
} 
} /* Output: 
Cookie constructor 
| ~ 


就 可 以 创建 一 个 Cookie 对 象 ， 因 为 它 的 构造 器 是 public 而 且 类 也 是 public 的 。( 此 后 我 们 将 会 
对 public 类 的 概念 了 解 更 多 。) 但 是 ， 由 于 bite0 只 向 在 dessert 包 中 的 类 提供 访问 权 ， 所 以 bite0 成 
员 在 Dinnerjava 之 中 是 无 法 访问 的 ， 因 此 编译 器 也 禁止 你 使 用 它 。 

默认 包 

令 人 吃惊 的 是 ， 下 面 的 程序 代码 虽然 看 起 来 破坏 了 上 述 规则 ， 但 它 仍 可 以 编译 : 

//: access/Cake. java 

// Accesses a class in a separate compilation unit. 


class Cake { 
public static void main(String[] args) { 
Pie x = new Pie(): 
«FO: 
y 
} /* Output: 
Pie. f() 
Whim 


在 第 二 个 处 于 相同 目录 的 文件 中 : 

//: access/Pie.java 

// The other class. 

class Pie { 

void f() { System.out.printin("Pie.f()"): } 

dh 

最 初 或 许 会 认为 这 两 个 文件 毫 不 相关 ， 但 Cake 却 可 以 创建 一 个 Pie 对 象 并 调用 它 的 f0 方 法 ! 
( 记 住 ， 为 了 使 文件 可 以 被 编译 ， 在 你 的 CLASSPATH 之 中 一 定 要 有 “.”。) 通常 会 认为 Pie 和 f0 享 
有 包 访问 权限 ， 因 而 是 不 可 以 为 Cake 所 用 的 。 它 们 的 确 享 有 包 访问 权限 ， 但 这 只 是 部 分 正确 的 。 
Cakejava 可 以 访问 它们 的 原因 是 因为 它们 同 处 于 相同 的 目录 并 且 没有 给 自己 设 定 任何 包 名 称 。 
Java 将 这 样 的 文件 自动 看 作 是 隶属 于 该 目录 的 默认 包 之 中 ， 于 是 它们 为 该 目录 中 所 有 其 他 的 文 
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件 都 提供 了 包 访 问 权 限 。 
6.2.3 private: 你 无 法 访问 

关键 字 private 的 意思 是 ， 除 了 包含 该 成 员 的 类 之 外 ， 其 他 任何 类 都 无 法 访问 这 个 成 员 。 由 
于 处 于 同一 个 包 内 的 其 他 类 是 不 可 以 访问 private 成 员 的 ， 因 此 这 等 于 说 是 自己 隔离 了 自己 。 从 
另 一 方面 说 ， 让 许多 人 共同 合作 来 创建 一 个 包 也 是 不 大 可 能 的 ， 为 此 private 就 允许 你 随意 改变 
该 成 员 ， 而 不 必 考 虑 这 样 做 是 否 会 影响 到 包 内 其 他 的 类 。 

默认 的 包 访问 权限 通常 已 经 提供 了 充足 的 隐藏 措施 。 请 记 住 ， 使 用 类 的 客户 端 程序 员 是 无 
法 访问 包 访问 权限 成 员 的 。 这 样 做 很 好 ， 因 为 默认 访问 权限 是 一 种 我 们 常用 的 权限 ， 同 时 也 是 
一 种 在 忘记 添加 任何 访问 权限 控制 时 能 够 自动 得 到 的 权限 。 因 此 ， 通 常 考虑 的 是 ， 哪 些 成 员 是 
想 要 明确 公开 给 客户 端 程序 员 使 用 的 ， 从 而 将 它们 声明 为 publice， 而 在 最 初 ， 你 可 能 不 会 认为 自 
己 经 常会 需要 使 用 关键 字 private， 因 为 没有 它 ， 照 样 可 以 工作 。 然 而 ， 事 实 很 快 就 会 证 明 ， 对 
private 的 使 用 是 多 么 的 重要 ， 在 多 线程 环境 下 更 是 如 此 (正如 将 在 第 21 章 中 看 到 的 )。 

此 处 有 一 个 使 用 private 的 示例 。 


//: access/IceCream. java 
// Demonstrates "private" keyword. 


class Sundae { 


private Sundae() {} 
static Sundae makeASundae() { 
return new Sundae(); 
} 
} 


public class IceCream { 
Public static void main(String{] args) { 
//! Sundae x = new Sundae(); 
Sundae x = Sundae.makeASundae(); 


} 

dM 

这 是 一 个 说 明 private 终 有 其 用 武之 地 的 示例 ， 可 能 想 控制 如 何 创建 对 象 ， 并 阻止 别人 直接 
访问 某 个 特定 的 构造 器 (或 全 部 构造 器 )。 在 上 面 的 例子 中 , 不 能 通过 构造 器 来 创建 Sundae 对 象 ， 
而 必须 调用 makeASundae0 方 法 来 达到 此 目的 9 。 

任何 可 以 肯定 只 是 该 类 的 一 个 “助手 ”方法 的 方法 ， 都 可 以 把 它 指 定 为 private， 以 确保 不 
会 在 包 内 的 其 他 地 方 误 用 到 它 ， 于 是 也 就 防止 了 你 会 去 改变 或 删除 这 个 方法 。 将 方法 指定 为 
Private 确保 了 你 拥有 这 种 选择 权 。 

这 对 于 类 中 的 private 域 同样 适用 。 除 非 必须 公开 底层 实现 细 目 (此 种 境况 很 少见 )， 否 则 就 
应 该 将 所 有 的 域 指定 为 private。 然 而 ， 不 能 因为 在 类 中 某 个 对 象 的 引用 是 private， 就 认为 其 他 
的 对 象 无 法 拥有 该 对 象 的 public 引用 (参见 本 书 的 在 线 补充 材料 以 了 解 有 关 别 名 机 制 的 话题 ) 。 
6.2.4 protected: 继承 访问 权限 

要 理解 protected 的 访问 权限 ， 我 们 在 内 容 上 需要 作 一 点 跳跃 。 首 先 ， 在 本 书 介绍 继承 (第 7 
章 ) 之 前 ， 读 者 并 不 需 真正 理解 本 节 的 内 容 。 但 为 了 内 容 的 完整 性 ， 这 里 还 是 提供 了 一 个 简要 
介绍 和 使 用 protected 的 示例 。 

关键 字 protected 处 理 的 是 继承 的 概念 ， 通 过 继承 可 以 利用 一 个 现 有 类 -我 们 将 其 称 为 基 类 ， 


O 此 例 还 有 另 一 个 效果 : 既然 于 认 构造 器 是 唯一 定义 的 构造 器 ， 并 且 它 是 private 的 ， 那 么 它 将 阻碍 对 此 类 的 继 
承 (我 们 将 在 后 面 介绍 这 个 问题 )。 
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然后 将 新 成 员 添 加 到 该 现 有 类 中 而 不 必 碰 该 现 有 类 。 还 可 以 改变 该 类 的 现 有 成 员 的 行为 。 为 了 
从 现 有 类 中 继承 ， 需 要 声明 新 类 extends (扩展 ) 了 一 个 现 有 类 ， 就 像 这 样 : 

class Foo extends Bar { 

类 定义 中 的 其 他 部 分 看 起 来 都 是 一 样 的 。 

如 果 创 建 了 一 个 新 包 ， 并 自 另 一 个 包 中 继承 类 ， 那 么 唯一 可 以 访问 的 成 员 就 是 源 包 的 public 
成 员 。( 当 然 ， 如 果 在 同一 个 包 内 执行 继承 工作 ， 就 可 以 操纵 所 有 的 拥有 包 访问 权限 的 成 员 。) 
有 时 ， 基 类 的 创建 者 会 希望 有 某 个 特定 成 员 ， 把 对 它 的 访问 权限 赋予 派生 类 而 不 是 所 有 类 。 这 
就 需要 protected 来 完成 这 一 工作 。protected 也 提供 包 访问 权限 ， 也 就 是 说 ， 相 同 包 内 的 其 他 类 
可 以 访问 protected 元 素 。 

回顾 一 下 先前 的 例子 Cookie.java， 就 可 以 得 知 下 面 的 类 是 不 可 以 调用 拥有 包 访问 权限 的 成 
员 bite0 的 : 


//: access/ChocolateChip. java 
// Can't use package-access member from another package. 
import access. dessert. *; 


public class ChocolateChip extends Cookie { 
public ChocolateChip() { 
System.out.printin("ChocolateChip constructor”); 


} 
public void chomp() { 
//! bite(); // Can't access bite 


public static void main(String[] args) { 
ChocolateChip x = new ChocolateChip(); 
x. chomp() ; 


} 
} /* Output: 
Cookie constructor 
ChocolateChip constructor 
Whim 


有 关 继承 技术 的 一 个 很 有 趣 的 事情 是 ， 如 果 类 Cookie 中 存在 一 个 方法 bite0 的 话 ， 那 么 该 方 
法 同时 也 存在 于 任何 一 个 从 Cookie 继 承 而 来 的 类 中 。 但 是 由 于 bite0 有 包 访 问 权限 而 且 它 位 于 另 
一 个 包 内 ， 所 以 我 们 在 这 个 包 内 是 无 法 使 用 它 的 。 当 然 ， 也 可 以 把 它 指 定 为 public， 但 是 这 样 做 
所 有 的 人 就 都 有 了 访问 权限 ， 而 且 很 可 能 这 并 不 是 你 所 希望 的 。 如 果 我 们 将 类 Cookie 像 这 样 加 
以 更 改 : 


//: access/cookie2/Cookie. java 
package access. cookie2; 


public class Cookie { 
public Cookie() { 
System. out.printin("Cookie constructor"); 
} 
protected void bite() { 
System.out.printin("bite"); 


} 
v= 


现在 bite0 对 于 所 有 继承 自 Cookie 的 类 而 言 ， 也 是 可 以 使 用 的 。 


//: access/ChocolateChip2. java 
‘import access.cookie2.*; 


public class ChocolateChip2 extends Cookie { 
public ChocolateChip2() { 
System.out.printin("ChocolateChip2 constructor"); 











226| 














120 _ KOR 





public void chomp() { bite(); } // Protected method 

public static void main(String{] args) { 
ChocolateChip2 x = new ChocolateChip2(); 
x.chomp() ; 


} 
} /* Output: 
Cookie constructor 
ChocolateChip2 constructor 
bite 
Wh 


注意 ， 尽 管 bite0 也 具有 包 访 问 权限 ， 但 是 它 仍旧 不 是 publie 的 。 

练习 4: (2) 展示 protected 方 法 具有 包 访问 权限 ， 但 不 是 public。 

练习 5: (2) 创建 一 个 带 有 public, private, protected 和 包 访 问 权限 域 以 及 方法 成 员 的 类 。 创 建 
该 类 的 一 个 对 象 ， 看 看 在 你 试图 调用 所 有 类 成 员 时 ， 会 得 到 什么 类 型 的 编译 信息 。 请 注意 ， 处 
于 同一 个 目录 中 的 所 有 类 都 是 默认 包 的 一 部 分 。 

练习 6: (1) 创建 一 个 带 有 protected 数 据 的 类 。 运 用 在 第 一 个 类 中 处 理 protected 数 据 的 方法 
在 相同 的 文件 中 创建 第 二 个 类 。 


6.3 接口 和 实现 


访问 权限 的 控制 常 被 称 为 是 具体 实现 的 隐藏 。 把 数据 和 方法 包装 进 类 中 ， 以 及 具体 实现 的 
隐藏 ， 常 共同 被 称 作 是 寺 装 8 。 其 结果 是 一 个 同时 带 有 特征 和 行为 的 数据 类 型 。 

出 于 两 个 很 重要 的 原因 ， 访 问 权限 控制 将 权限 的 边界 划 在 了 数据 类 型 的 内 部 。 第 一 个 原因 
是 要 设 定 客户 端 程序 员 可 以 使 用 和 不 可 以 使 用 的 界限 。 可 以 在 结构 中 建立 自己 的 内 部 机 制 ， 而 
不 必 担 心 客户 端 程序 员 会 偶然 地 将 内 部 机 制 当 作 是 他 们 可 以 使 用 的 接口 的 一 部 分 。 

这 个 原因 直接 引出 了 第 二 个 原因 ， 即 将 接口 和 具体 实现 进行 分 离 。 如 果 结构 是 用 于 一 组 程 
序 之 中 ， 而 客户 端 程序 员 除了 可 以 向 接口 发 送信 息 之 外 什么 也 不 可 以 做 的 话 ， 那 么 就 可 以 随意 
更 改 所 有 不 是 public 的 东西 (例如 有 包 访问 权限 、protected 和 private 的 成 员 ) ， 而 不 会 破坏 客 
户 端 代码 。 

为 了 清楚 起 见 ， 可 能 会 采用 一 种 将 public 成 员 置 于 开头 ， 后 面 跟着 protected、 包 访问 权限 和 
private 成 员 的 创建 类 的 形式 。 这 样 做 的 好 处 是 类 的 使 用 者 可 以 从 头 读 起 ， 首 先 阅 读 对 他 们 而 言 
最 为 重要 的 部 分 〈 即 publie 成 员 ， 因 为 可 以 从 文件 外 部 调用 它们 ) ， 等 到 遇见 作为 内 部 实现 细节 
的 非 public 成 员 时 停止 阅读 : 


//: access/OrganizedByAccess.java 


public class OrganizedByAccess { 
public void publ() { /* * 
public void pub2() { /* 
public void pub3() { /* 
Private void privl() { 
Private void priv2() { /* ... 
private void priv3() { /* ... * 
private int i; 
II vas 

} Ii~ 


这 样 做 仅 能 使 程序 阅读 起 来 稍微 容易 一 些 ， 因 为 接口 和 具体 实现 仍旧 混在 一 起 。 也 就 是 说 ， 
仍 能 看 到 源 代码 一 实现 部 分 ， 因 为 它 就 在 类 中 。 另 外 ，javadoc 所 提供 的 注释 文档 功能 降低 了 程 






序 代码 的 可 读 性 对 客户 端 程序 员 的 重要 性 。 将 接口 展现 给 某 个 类 的 使 用 者 实际 上 是 类 浏览 器 的 


O 然而 ， 人 们 经 常 只 单独 将 具体 实现 的 隐藏 称 作 封装 。 
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任务 。 类 浏览 器 是 一 种 以 非常 有 用 的 方式 来 查阅 所 有 可 用 的 类 ， 并 告诉 你 用 它们 可 以 做 些 什么 
(也 就 是 显示 出 可 用 成 员 ) 的 工具 。 在 Java 中 ， 用 Web 浏 览 器 浏览 JDK 文 档 可 以 得 到 使 用 类 浏览 
器 的 相同 效果 。 


6.4 类 的 访问 权限 


在 Java 中 ， 访 问 权限 修饰 词 也 可 以 用 于 确定 库 中 的 哪些 类 对 于 该 库 的 使 用 者 是 可 用 的 。 如 
果 希 望 某 个 类 可 以 为 某 个 客户 端 程序 员 所 用 ， 就 可 以 通过 把 关键 字 public 作 用 于 整个 类 的 定义 来 
达到 目的 。 这 样 做 甚至 可 以 控制 客户 端 程序 员 是 否 能 创建 一 个 该 类 的 对 象 。 

为 了 控制 某 个 类 的 访问 权限 ， 修 饰 词 必须 出 现 于 关键 字 class 之 前 。 因 此 可 以 像 下 面 这 样 声明 ; 

public class Widget { 

现在 如 果 库 的 名 字 是 access， 那 么 任何 客户 端 程序 员 都 可 以 通过 下 面 的 声明 访问 Widget; 

import access.Widget; 

或 

import access.*; 

然而 ， 这 里 还 有 一 些 额外 的 限制 : 

1. 每 个 编译 单元 (文件) 都 只 能 有 一 个 public 类 。 这 表示 ， 每 个 编译 单元 都 有 单一 的 公共 接 
口 ， 用 public 类 来 表现 。 该 接口 可 以 按 要 求 包含 众多 的 支持 包 访问 权限 的 类 。 如 果 在 某 个 编译 单 
元 内 有 一 个 以 上 的 public 类 ， 编 译 器 就 会 给 出 出 错 信息 。 

2, public 类 的 名 称 必须 完全 与 含有 该 编译 单元 的 文件 名 相 匹配 ， 包 括 大 小 写 。 所 以 对 于 
Widget 而 言 ， 文 件 的 名 称 必 须 是 Widgetjava， 而 不 是 widgetjava 或 WIDGETjava。 如 果 不 匹 配 ， 
同样 将 得 到 编译 时 错误 。 

3. 虽然 不 是 很 常用 ， 但 编译 单元 内 完全 不 带 public 类 也 是 可 能 的 。 在 这 种 情况 下 ， 可 以 随意 
对 文件 命名 。( 尽 管 随意 命名 会 使 得 人 们 在 阅读 和 维护 代码 时 产生 混淆 。) 

如 果 获 取 了 一 个 在 access 内 部 的 类 ， 用 来 完成 Widget 或 是 其 他 在 access 中 的 public 类 所 要 执 
行 的 任务 ， 将 会 出 现 什么 样 的 情况 呢 ? 你 不 想 自 找 麻烦 去 为 客户 端 程序 员 创建 说 明文 档 ， 而 且 
你 认为 不 久 可 能 会 想 要 完全 改变 原 有 方案 并 将 旧版 本 一 起 删除 ， 代 之 以 一 种 不 同 的 版 本 。 为 了 
保留 此 灵活 性 ， 需 要 确保 客户 端 程序 员 不 会 依赖 于 隐藏 在 access 之 中 的 任何 特定 实现 细节 。 为 了 
达到 这 一 点 ， 只 需 将 关键 字 public 从 类 中 拿 掉 ， 这 个 类 就 拥有 了 包 访问 权限 。( 该 类 只 可 以 用 于 


该 包 之 中 。) 
练习 7: (1) 根据 描述 access 和 Widget 的 代码 片段 创建 类 库 。 在 某 个 不 属于 access 类 库 的 类 中 
创建 一 个 Widget 实 例 。 


在 创建 一 个 包 访问 权限 的 类 时 ， 仍 旧 是 在 将 该 类 的 域 声明 为 private 时 才 有 意义 一 应 尽 可 能 地 
总 是 将 域 指定 为 私有 的 ， 但 是 通常 来 说 ， 将 与 类 〈 包 访问 权限 ) 相同 的 访问 权限 赋予 方法 也 是 
很 合理 的 。 既 然 一 个 有 包 访 问 权限 的 类 通常 只 能 被 用 于 包 内 ， 那 么 如 果 对 你 有 强制 要 求 ， 在 此 
种 情况 下， 编译 器 会 告诉 你 ， 你 只 需要 将 这 样 的 类 的 方法 设 定 为 public 就 可 以 了 。 

请 注意 ， 类 既 不 可 以 是 private 的 〈 这 样 会 使 得 除 该 类 之 外 ， 其 他 任何 类 都 不 可 以 访问 它 )， 
也 不 可 以 是 protected 的 8 。 所 以 对 于 类 的 访问 权限 ， 仅 有 两 个 选择 : 包 访问 权限 或 public。 如 果 
不 希望 其 他 任何 人 对 该 类 拥有 访问 权限 ， 可 以 把 所 有 的 构造 器 都 指定 为 private， 从 而 阻止 任何 人 
创建 该 类 的 对 象 ， 但 是 有 一 个 例外 ， 就 是 你 在 该 类 的 static 成 员 内 部 可 以 创建 。 下 面 是 一 个 示例 : 


O 事实 上 ， 一 个 内 部 类 可 以 是 private 或 是 protected 的 ， 但 那 是 一 个 特例 。 这 将 在 第 10 章 中 介绍 到 。 
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/1/: access/Lunch. java 
// Demonstrates class access specifiers. Make a class 
// effectively private with private constructors: 


class Soupl { 
private Soupl() {} 
7/ (1) Allow creation via static method: 
public static Soupl makeSoup() { 
return new Soupl(); 
} 
} 


class Soup2 { 
private Soup2() {} 
7/ (2) Create a static object and return a reference 
7/ upon request. (The "Singleton" pattern): 
private static Soup2 psi = new Soup2(): 
public static Soup2 access() { 
return psl; 


$ 
public void f() {} 
) 


// Only one public class allowed per file: 
public class Lunch { 
void testPrivate() { 
// Can't do this! Private constructor: 
/1/! Soup soup = new Soup1(); 


void testStatic() { 
Soup1 soup = Soupl.makeSoup(); 
} 
void testSingleton() { 
Soup2.access().#(); 


} 
} Ii~ 
到 目前 为 止 ， 绝 大 多 数 方法 均 返 回 void 或 基本 类 型 ， 所 以 定义 


public static Soupl makeSoup() { 
return new Soupl(); 


} 


初 看 起 来 可 能 有 点 令 人 迷惑 不 解 。 方 法 名 称 (makeSoup) 前 面 的 词 Soup1 告 知 了 该 方法 返 
回 的 东西 。 本 书 到 目前 为 止 ， 这 里 经 常 是 void， 意 思 是 它 不 返回 任何 东西 。 但 是 也 可 以 返回 一 
个 对 象 引 用 ， 示 例 中 就 是 这 种 情况 。 这 个 方法 返回 了 一 个 对 Soup1 类 的 对 象 的 引用 。 

Soup1 类 和 Soup2 类 展示 了 如 何 通过 将 所 有 的 构造 器 指定 为 private 来 阻止 直接 创建 某 个 类 
的 实例 。 请 一 定 要 牢记 ， 如 果 没 有 明确 地 至 少 创建 一 个 构造 器 的 话 ， 就 会 帮 你 创建 一 个 默认 构 
造 器 (不 带 有 任何 参数 的 构造 器 ) 。 如 果 我 们 自己 编写 了 默认 的 构造 器 ， 那 么 就 不 会 自动 创建 
它 了 。 如 果 把 该 构造 器 指定 为 private， 那 么 就 谁 也 无 法 创建 该 类 的 对 象 了 。 但 是 现在 别人 该 怎 
样 使 用 这 个 类 呢 ? 上 面 的 例子 就 给 出 了 两 种 选择 : 在 Soup1 中 ， 创 建 一 个 static 方 法 ， 它 创建 一 
个 新 的 Soup1 对 象 并 返回 一 个 对 它 的 引用 。 如 果 想 要 在 返回 引用 之 前 在 Soupl 上 做 一 些 额外 的 
工作 ， 或 是 如 果 想 要 记录 到 底 创建 了 多 少 个 Soup1 对 象 〔 可 能 要 限制 其 数量 )， 这 种 做 法 将 会 是 
大 有 神 益 的 。 

Soup2 用 到 了 所 谓 的 设计 模式 ， 该 模式 在 www.MindView.net 网 站 《Thinking in Patterns (with 
Java)》 一 书 中 有 所 介绍 。 这 种 特定 的 模式 被 称 为 singleton( 单 例 )， 这 是 因为 你 始终 只 能 创建 它 的 
一 个 对 象 。Soup2 类 的 对 象 是 作为 Soup2 的 一 个 static private 成 员 而 创建 的 ， 所 以 有 且 仅 有 一 个 ， 
而 且 除 非 是 通过 public 方 法 access0， 否 则 是 无 法 访问 到 它 的 。 
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正如 前 面 所 提 到 的 ， 如 果 没 能 为 类 访问 权限 指定 一 个 访问 修饰 符 ， 它 就 会 默认 得 到 包 访问 
权限 。 这 就 意味 着 该 类 的 对 象 可 以 由 包 内 任何 其 他 类 来 创建 ， 但 在 包 外 则 是 不 行 的 。( 一 定 要 记 
住 ， 相 同 目录 下 的 所 有 不 具有 明确 package 声 明 的 文件 ， 都 被 视 作 是 该 目录 下 默认 包 的 一 部 分 。) 
然而 ， 如 果 该 类 的 某 个 statie 成 员 是 public 的 话 ， 则 客户 端 程序 员 仍旧 可 以 调用 该 static 成 员 ， 尽 
管 他 们 并 不 能 生成 该 类 的 对 象 。 

练习 8: (4) 效仿 示例 Lunchjava 的 形式 ， 创 建 一 个 名 为 ConnectionManager 的 类 ， 该 类 管理 
一 个 元 素 为 Connection 对 象 的 固定 数组 。 客 户 端 程序 员 不 能 直接 创建 Connection 对 象 ， 而 只 能 
通过 ConnectionManager 中 的 某 个 static 方 法 来 获取 它们 。 当 ConnectionManager 之 中 不 再 有 对 
象 时 ， 它 会 返回 null 引 用 。 在 main0 之 中 检测 这 些 类 。 

练习 9:，(2) 在 accessy/local 目 录 下 编写 以 下 文件 (假定 access/local 目 录 在 你 的 CLASSPATH 中 ): 

// access/local/PackagedClass.java 

package access. local; 


class PackagedClass { 
public PackagedClass() { 
System.out.printin("Creating a packaged class"); 
} 
然后 在 accesss/local 之 外 的 另 一 个 目录 中 创建 下 列 文件 : 


// access/foreign/Foreign. java 
package access. foreign; 
import access. local.*; 


public class Foreign { 
public static void main(String{] args) { 
PackagedClass pc = new PackagedClass(); 


} 
} 


解释 一 下 为 什么 编译 器 会 产生 错误 。 如 果 将 Foreign 类 置 于 access.local 包 之 中 的 话 ， 会 有 所 
改变 吗 ? 


6.5 总 结 


无 论 是 在 什么 样 的 关系 之 中 ， 设 立 一 些 为 各 成 员 所 遵守 的 界限 始终 是 很 重要 的 。 当 创建 了 
一 个 类 库 ， 也 就 与 该 类 库 的 用 户 建立 了 某 种 关系 ， 这 些 用 户 就 是 客户 端 程序 员 ， 他 们 是 另外 一 
些 程序 员 ， 他 们 将 你 的 类 库 聚 合成 为 一 个 应 用 程序 ， 或 是 运用 你 的 类 库 来 创建 一 个 更 大 的 类 库 。 

如 果 不 制定 规则 ， 客 户 端 程序 员 就 可 以 对 类 的 所 有 成 员 随 心 而 为 ， 即 使 你 可 能 并 不 希望 他 
们 直接 复制 其 中 的 一 些 成 员 。 在 这 种 情况 下 ， 所 有 事物 都 是 公开 的 。 

本 章 讨 论 了 类 是 如 何 被 构建 成 类 库 的 : 首先 ， 介 绍 了 一 组 类 是 如 何 被 打包 到 一 个 类 库 中 
的 ， 其次， 类 是 如 何 控制 对 其 成 员 的 访问 的 。 

据 估计 ， 用 C 语 言 开发 项 目 ， 在 50 千 行 至 100 千 行 代码 之 间 就 会 出 现 问题 。 这 是 因为 C 语 言 
仅 有 单一 的 “名 字 空间 "， 并 且 名 称 开始 发 生 冲突 ， 引 发 额外 的 管理 开销 。 而 对 于 Java， 关 键 字 
Package、 包 的 命名 模式 和 关键 字 import， 可 以 使 你 对 名 称 进行 完全 的 控制 ， 因 此 名 称 冲突 的 问 
题 是 很 容易 避免 的 。 

控制 对 成 员 的 访问 权限 有 两 个 原因 。 第 一 是 为 了 使 用 户 不 要 磁 触 那些 他 们 不 该 磁 触 的 部 分 ， 
这 些 部 分 对 于 类 内 部 的 操作 是 必要 的 ， 但 是 它 并 不 属于 客户 端 程序 员 所 需 接口 的 一 部 分 。 因 此 ， 
将 方法 和 域 指定 成 private， 对 客户 端 程序 员 而 言 是 一 种 服务 。 因 为 这 样 他 们 可 以 很 清楚 地 看 到 
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什么 对 他 们 重要 ， 什 么 是 他 们 可 以 忽略 的 。 这 样 简化 了 他 们 对 类 的 理解 。 

第 二 个 原因 ， 也 是 最 重要 的 原因 ， 是 为 了 让 类 库 设计 者 可 以 更 改 类 的 内 部 工作 方式 ， 而 不 
必 担 心 这 样 会 对 客户 端 程序 员 产生 重大 的 影响 。 例 如 ， 最 初 可 能 会 以 某 种 方式 创建 一 个 类 ， 然 
后 发 现 如 果 更 改 程序 结构 ， 可 以 大 大 提高 运行 速度 。 如 果 接口 和 实现 可 以 被 明确 地 隔离 和 加 以 
保护 ， 那 么 就 可 以 实现 这 一 目的 ， 而 不 必 强 制 客户 端 程序 员 重新 编写 代码 。 访 问 权限 控制 可 以 
确保 不 会 有 任何 客户 端 程序 员 依赖 于 某 个 类 的 底层 实现 的 任何 部 分 。 

当 具 备 了 改变 底层 实施 细节 的 能 力 时 ， 不 仅 可 以 随意 地 改善 设计 ， 还 可 能 会 随意 地 犯错 误 
同时 也 就 有 了 犯错 的 可 能 性 。 无 论 如 何 细心 地 计划 并 设计 ， 都 有 可 能 犯错 。 当 了 解 到 你 所 犯错 
误 是 相对 安全 的 时 候 ， 就 可 以 更 加 放心 地 进行 实验 ， 也 就 可 以 更 快 地 学 会 ， 更 快 地 完成 项 目 。 

类 的 公共 接口 是 用 户 真正 能 够 看 到 的 ， 所 以 这 一 部 分 是 在 分 析 和 设计 的 过 程 中 决定 该 类 是 
否 正确 的 最 重要 的 部 分 。 尽 管 如 此 ， 你 仍然 有 进行 改变 的 空间 。 如 果 在 最 初 无 法 创建 出 正确 的 
接口 ， 那 么 只 要 不 删除 任何 客户 端 程序 员 在 他 们 的 程序 中 已 经 用 到 的 东西 ， 就 可 以 在 以 后 添加 
更 多 的 方法 。 

注意 ,访问 权限 控制 专注 于 类 库 创建 者 和 该 类 库 的 外 部 使 用 者 之 间 的 关系 ， 这 种 关系 也 是 
一 种 通信 方式 。 然 而 ， 在 许多 情况 下 事情 并 非 如 此 。 例 如 ， 你 自己 编写 了 所 有 的 代码 ， 或 者 你 
在 一 个 组 员 聚 集 在 一 起 的 项 目 组 中 工作 ， 所 有 的 东西 都 放 在 同一 个 包 中 。 这 些 情 况 是 另外 一 种 
不 同 的 通信 方式 ， 因 此 严格 地 遵循 访问 权限 规则 并 不 一 定 是 最 佳 选择 ， 默 认 ( 包 ) 访问 权限 也 
许 只 是 可 行 而 已 。 

所 选 习题 的 答案 都 可 以 在 名 为 The Thinking in Java Annotated Solution Guide 的 电子 文档 中 找 
到 ， 读 者 可 以 从 www.MindView.net 购 买 此 文档 。 





第 7 章 复 用 类 


复 用 代码 是 Java 众 多 引 人 注 目的 功能 之 一 。 但 要 想 成 为 极 具 革命 性 的 语言 ， 仅 仅 能 够 复制 
代码 并 对 之 加 以 改变 是 不 够 的 ， 它 还 必须 能 够 做 更 多 的 事情 。 

上 述 方法 常 为 C 这 类 过 程 型 语言 所 使 用 ， 但 收效 并 不 是 很 好 。 正 如 Java 中 所 有 事物 一 样 ， 问 
题解 决 都 是 围绕 着 类 展开 的 。 可 以 通过 创建 新 类 来 复 用 代码 ， 而 不 必 再 重头 开始 编写 。 可 以 使 
用 别人 业已 开发 并 调试 好 的 类 。 

此 方法 的 窍门 在 于 使 用 类 而 不 破坏 现 有 程序 代码 。 读 者 将 会 在 本 章 中 看 到 两 种 达到 这 一 目 
的 的 方法 。 第 一 种 方法 非常 直观 ， 只 需 在 新 的 类 中 产生 现 有 类 的 对 象 。 由 于 新 的 类 是 由 现 有 
类 的 对 象 所 组 成 ， 所 以 这 种 方法 称 为 组 合 。 该 方法 只 是 复 用 了 现 有 程序 代码 的 功能 ， 而 非 它 
的 形式 。 

第 二 种 方法 则 更 细致 一 些 ， 它 按照 现 有 类 的 类 型 来 创建 新 类 。 无 需 改 变现 有 类 的 形式 ， 采 
用 现 有 类 的 形式 并 在 其 中 添加 新 代码 。 这 种 神奇 的 方式 称 为 继承 ， 而 且 编译 器 可 以 完成 其 中 大 
部 分 工作 。 继 承 是 面向 对 象 程序 设计 的 基石 之 一 ， 我 们 将 在 第 8 章 中 探究 其 含义 与 功能 。 

就 组 合 和 继承 而 言 ， 其 语法 和 行为 大 多 是 相似 的 。 由 于 它们 是 利用 现 有 类 型 生成 新 类 型 ， 
所 以 这 样 做 极 富 意义 。 在 本 章 中 ， 读 者 将 会 了 解 到 这 两 种 代码 重用 机 制 。 


7.1 组 合 语法 


本 书 到 目前 为 止 ， 已 多 次 使 用 组 合 技术 。 只 需 将 对 象 引用 置 于 新 类 中 即 可 。 例 如 ， 假 设 你 
需要 某 个 对 象 ， 它 要 具有 多 个 String 对 象 、 几 个 基本 类 型 数据 ， 以 及 另 一 个 类 的 对 象 。 对 于 非 基 
本 类 型 的 对 象 ， 必 须 将 其 引用 置 于 新 的 类 中 ， 但 可 以 直接 定义 基本 类 型 数据 : 


//: reusing/SprinklerSystem. java 
// Composition for code reuse. 


class WaterSource { 
private String s: 
WaterSource() { 
System.out.printin("WaterSource()") ; 
s = "Constructed"; 


} 
Public String toString() { return s; } 


public class SprinklerSystem { 
private String valvel, valve2, valve3, valved; 
private WaterSource source = new WaterSource(): 
private int i; 
private float f; 
public String toString() { 
return 
"valvel = " + valvel + * + 
"valve? = * + valved +" "+ 
“valve3 = " + valved + * + 
"valved = " + valved + "\n" + 
Po a 2-9 9 aS AGF: He BR ey 
“source = " + source; 7 











|239] 








126 RIE 





public static void main(String{] args) { 
SprinklerSystem sprinklers = new SprinklerSystem(): 
System.out.printin (sprinklers); 


} 
} /* Output: 
WaterSource() 
valvel = null valve2 = null valve3 = null valved = null 
i = @ f = @.@ source = Constructed 
Whim 


在 上 面 两 个 类 所 定义 的 方法 中 ， 有 一 个 很 特殊 : toString0。 每 一 个 非 基本 类 型 的 对 象 都 有 
一 个 toString0 方 法 ， 而 且 当 编译 器 需要 一 个 String 而 你 却 只 有 一 个 对 象 时 ， 该 方法 便 会 被 调用 。 
所 以 在 SprinklerSystem.toString0 的 表达 式 中 : 

“source = " + source; 
HAVES HA UAB — 7 Stringxt RX (“source=") 同 WaterSource 相 加 。 由 于 只 能 将 一 个 
String 对 象 和 另 一 个 String 对 象 相 加 ， 因 此 编译 器 会 告诉 你 :“ 我 将 调用 toString0， 把 source 转 换 
成 为 一 个 String! ”这 样 做 之 后 ， 它 就 能 够 将 两 个 String 连 接 到 一 起 并 将 结果 传递 给 System.out 
sprinting (或 者 使 用 与 此 等 价 的 本 书 中 静态 的 print0 和 printnb0 方 法 )。 每 当 想 要 使 所 创建 的 类 具 
备 这 样 的 行为 时 ， 仅 需要 编写 一 个 toString0 方 法 即 可 。 

正如 我 们 在 第 2 章 中 所 提 到 的 ， 类 中 域 为 基本 类 型 时 能 够 自动 被 初始 化 为 零 。 但 是 对 象 引用 
会 被 初始 化 为 aull， 而 且 如 果 你 试图 为 它们 调用 任何 方法 ， 都 会 得 到 一 个 异常 一 一 运行 时 错误 。 
很 方便 的 是 ， 在 不 抛 出 异常 的 情况 下 仍旧 可 以 打印 一 个 null 引 用 。 

编译 器 并 不 是 简单 地 为 每 一 个 引用 都 创建 默认 对 象 ， 这 一 点 是 很 有 意义 的 ， 因 为 若 真 要 那 
样 做 的 话 ， 就 会 在 许多 情况 下 增加 不 必要 的 负担 。 如 果 想 初始 化 这 些 引 用 ， 可 以 在 代码 中 的 下 
列 位 置 进行 ; 

1, 在 定义 对 象 的 地 方 。 这 意味 着 它们 总 是 能 够 在 构造 器 被 调用 之 前 被 初始 化 。 

2. 在 类 的 构造 器 中 。 

3. 就 在 正 要 使 用 这 些 对 象 之 前 ， 这 种 方式 称 为 情 性 初始 化 。 在 生成 对 象 不 值得 及 不 必 每 次 
都 生成 对 象 的 情况 下 ， 这 种 方式 可 以 减少 额外 的 负担 。 

4. 使 用 实例 初始 化 。 

以 下 是 这 四 种 方式 的 示例 : 


/1: reusing/Bath. java 
// Constructor initialization with composition. 
import static net.mindview.util.Print.*; 


class Soap { 
private String s; 
Soap() { 
print("Soap()"): 
s = "Constructed"; 


站 
public String toString() { return s; } 
} 


public class Bath { 

private String // Initializing at point of definition: 
sl = "Happy", 
s2 = "Happy", 
53, s4; 

private Soap castille; 

private int i; 

private float toy; 

public Bath() { 
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print(*Inside Bath()"); 
53 = "Joy"; 
toy = 3.14f; 
castille = new Soap(); 


/1 Instance initialization: 
{i= 47; } 
public String toString() { 
if(s4 == null) // Delayed initialization: 
s4 = "Joy"; 
return 
"s 





+ 
oes 
+ 


i 
“toy = "+ toy + "\n" + 
"castille = ”+ castille; 
} 
public static void main(String[] args) { 
Bath b = new Bath(); 
print(b); 


} 
} /* Output: 
Inside Bath() 
Soap() 
s1 = Happy 
s2 = Happy 
s3 = Joy 
s4 = Joy 
i= 47 
toy = 3.14 
castille = Constructed 
Whim 


请 注意 ， 在 Bath 的 构造 器 中 ， 有 一 行 语句 在 所 有 初始 化 产生 之 前 就 已 经 执行 了 。 如 果 没 有 
在 定义 处 初始 化 ， 那 么 除非 发 生 了 不 可 避免 的 运行 期 异常 ， 否 则 将 不 能 保证 信息 在 发 送 给 对 象 
引用 之 前 已 经 被 初始 化 。 

当 toString0 被 调用 时 ， 它 将 填充 s4 的 值 ， 以 确保 所 有 的 域 在 使 用 之 时 已 被 妥善 初始 化 。 

练习 1，(2) 创建 一 个 简单 的 类 。 在 第 二 个 类 中 ， 将 一 个 引用 定义 为 第 一 个 类 的 对 象 。 运 用 
情 性 初始 化 来 实例 化 这 个 对 象 。 


7.2 继承 语法 


继承 是 所 有 OOP 语 言 和 Java 语 言 不 可 缺少 的 组 成 部 分 。 当 创建 一 个 类 时 ， 总 是 在 继承 ， 因 
此 ， 除 非 已 明确 指出 要 从 其 他 类 中 继承 ， 否 则 就 是 在 隐 式 地 从 Java 的 标准 根 类 Object 进行 继承 。 

组 合 的 语法 比较 平实 ， 但 是 继承 使 用 的 是 一 种 特殊 的 语法 。 在 继承 过 程 中 ， 需 要 先 声明 
“新 类 与 旧 类 相似 "。 这 种 声明 是 通过 在 类 主体 的 左边 花 括号 之 前 ， 书 写 后 面 紧 随 基 类 名 称 的 关 
键 字 extends 而 实现 的 。 当 这 么 做 时 ， 会 自动 得 到 基 类 中 所 有 的 域 和 方法 。 例 如 : 


//: reusing/Detergent.java 
// Inheritance syntax & properties. 
import static net.mindview.util.Print.*; 


class Cleanser { 
private String s = "Cleanser"; 
public void append(String a) { s += a; } 
public void dilute() { append(" dilute()"); } 
public void apply() { append(" apply()"): 
public void scrub() { append(” scrub()"); } 
public String toString() { return s; } 
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public static void main(String{] args) { 
Cleanser x = new Cleanser(): 
x.dilute(); x-apply(): x.scrub(): 
print (x): 

} 


public class Detergent extends Cleanser { 


// Change a method: 

public void scrub() { 
append(" Detergent.scrub()"); 
super.scrub(); // Call base-class version 


71 Add methods to the interface: 
Public void foam() { append(" foam()"); } 
// Test the new class: 
public static void main(String{] args) { 
Detergent x = new Detergent(); 
x.dilute(); 
x.apply(); 
x.scrub(); 
x. foam() ; 
print (x); 
print("Testing base class:"); 
Cleanser .main(args) ; 


} ys Output: 

Cleanser dilute() apply() Detergent.scrub() scrub() foam() 

Testing base class: 

Graner dilute() apply() scrub() 

这 个 程序 示范 了 Java 的 许多 特性 。 首 先 ， 在 Cleanser 的 append0 方 法 中 ， 我 们 用 “+=” 操 
作 符 将 几 个 String 对 象 连接 成 s， 此 操作 符 是 被 Java 设 计 者 重 载 用 以 处 理 String 对 象 的 操作 符 之 一 
( 另 一 个 是 “+ )。 

其 次 ，Cleanser 和 Detergent 均 含有 main0 方 法 。 可 以 为 每 个 类 都 创建 一 个 main0 方 法 。 这 种 
在 每 个 类 中 都 设置 一 个 main() 方 法 的 技术 可 使 每 个 类 的 单元 测试 都 变 得 简便 易 行 。 而 且 在 完成 
单元 测试 之 后 ， 也 无 需 删 除 main0， 可 以 将 其 留待 下 次 测试 。 

即使 是 一 个 程序 中 含有 多 个 类 ， 也 只 有 命令 行 所 调用 的 那个 类 的 main0 方 法 会 被 调用 。 因 
此 ， 在 此 例 中 ， 如 果 命令 行 是 java Detergent， 那 么 Detergentmain0 将 会 被 调用 。 即 使 Cleanser 
不 是 一 个 public 类 ， 如 果 命令 行 是 java Cleanser， 那 么 Cleansermain0 仍然 会 被 调用 。 即 使 一 
个 类 只 具有 包 访问 权限 ， 其 public main0 仍 然 是 可 访问 的 。 

在 此 例 中 ， 可 以 看 到 Detergentmain0 明 确 调 用 了 Cleansermain0， 并 将 从 命令 行 获取 的 参 
数 传递 给 了 它 。 当 然 ， 也 可 以 向 其 传递 任意 的 String 数组 。 

Cleanser 中 所 有 的 方法 都 必须 是 public 的 ， 这 一 点 非常 重要 。 请 记 住 ， 如 果 没 有 加 任何 访问 
权限 修饰 词 ， 那 么 成 员 默 认 的 访问 权限 是 包 访问 权限 ， 它 仅 允 许 包 内 的 成 员 访问 。 因 此 ， 在 此 
包 中 ， 如 果 没 有 访问 权限 修饰 词 ， 任 何人 都 可 以 使 用 这 些 方法 。 例 如 ，Detergent 就 不 成 问题 。 
但 是 ， 其 他 包 中 的 某 个 类 若 要 从 Cleanser 中 继承 ， 则 只 能 访问 publie 成 员 。 所 以 ， 为 了 继承 ， 一 
般 的 规则 是 将 所 有 的 数据 成 员 都 指定 为 private， 将 所 有 的 方法 指定 为 public ( 稍 后 将 会 学 到 ， 
Protected 成 员 也 可 以 借助 导出 类 来 访问 )。 当 然 ， 在 特殊 情况 下 ， 必 须 做 出 调整 ， 但 上 述 方法 的 
确 是 一 个 很 有 用 的 规则 。 

在 Cleanser 的 接口 中 有 一 组 方法 : append0、dilute0、apply0、serub0 和 toString0。 由 于 
Detergent 是 由 关键 字 extends 从 Cleanser 导 出 的 ， 所 以 它 可 以 在 其 接口 中 自动 获得 这 些 方法 ， 尽 
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管 并 不 能 看 到 这 些 方法 在 Detergent 中 的 显 式 定义 。 因 此 ， 可 以 将 继承 视 作 是 对 类 的 复 用 。 

正如 我 们 在 secrubO 中 所 见 ， 使 用 基 类 中 定义 的 方法 及 对 它 进 行 修改 是 可 行 的 。 在 此 例 中 ， 
你 可 能 想 要 在 新 版 本 中 调用 从 基 类 继承 而 来 的 方法 。 但 是 在 serub0 中 ， 并 不 能 直接 调用 serub0， 
因为 这 样 做 将 会 产生 递归 ， 而 这 并 不 是 你 所 期 望 的 。 为 解决 此 问题 ，Java 用 super 关 键 字 表示 超 
类 的 意思 ， 当 前 类 就 是 从 超 类 继承 来 的 。 为 此 ， 表 达 式 superscrub0 将 调用 基 类 版 本 的 secrub0 
方法 。 

在 继承 的 过 程 中 ， 并 不 一 定 非得 使 用 基 类 的 方法 。 也 可 以 在 导出 类 中 添加 新 方法 ， 其 添加 
方式 与 在 类 中 添加 任意 方法 一 样 ， 即 对 其 加 以 定义 即 可 。foam0 方 法 即 为 一 例 。 

读者 在 Detergent.main0 中 会 发 现 ， 对 于 一 个 Detegent 对 象 而 言 ， 除 了 可 以 调用 Detergent 的 
方法 〈 即 foam0) 之 外 ， 还 可 以 调用 Cleanser 中 所 有 可 用 的 方法 。 

练习 2: (2) 从 Detergent 中 继承 产生 一 个 新 的 类 。 覆 盖 scrub0 并 添加 一 个 名 为 sterilize0 的 新 
方法 。 


7.2.1 初始 化 基 类 

由 于 现在 涉及 基 类 和 导出 类 这 两 个 类 ， 而 不 是 只 有 一 个 类 ， 所 以 要 试 着 想像 导出 类 所 产生 
的 结果 对 象 ， 会 有 点 困惑 。 从 外 部 来 看 ， 它 就 像 是 一 个 与 基 类 具有 相同 接口 的 新 类 ， 或 许 还 会 
有 一 些 额 外 的 方法 和 域 。 但 继承 并 不 只 是 复制 基 类 的 接口 。 当 创建 了 一 个 导出 类 的 对 象 时 ， 该 
对 象 包含 了 一 个 基 类 的 子 对 象 。 这 个 子 对 象 与 你 用 基 类 直接 创建 的 对 象 是 一 样 的。 二 者 区 别 在 
于 ， 后 者 来 自 于 外 部 ， 而 基 类 的 子 对 象 被 包装 在 导出 类 对 象 内 部 。 

当然 ， 对 基 类 子 对 象 的 正确 初始 化 也 是 至 关 重 要 的 ， 而 且 也 仅 有 一 种 方法 来 保证 这 一 点 ， 
在 构造 器 中 调用 基 类 构造 器 来 执行 初始 化 ， 而 基 类 构造 器 具有 执行 基 类 初始 化 所 需要 的 所 有 知 
识 和 能 力 。jJava 会 自动 在 导出 类 的 构造 器 中 插入 对 基 类 构造 器 的 调用 。 下 例 展示 了 上 述 机 制 在 
三 层 继承 关系 上 是 如 何 工作 的 : 


//: reusing/Cartoon. java 
// Constructor calls during inheritance. 
import static net.mindview.util.Print.*; 


class Art { 
Art() { print("Art constructor"); } 
} 


class Drawing extends Art { 
Drawing() { print(*Drawing constructor"); } 


public class Cartoon extends Drawing { 
public Cartoon() { print("Cartoon constructor"); } 
public static void main(String(] args) { 
Cartoon x = new Cartoon(); 


} 
} /* Output: 
Art constructor 
Drawing constructor 
Cartoon constructor 
H~ 


读者 会 发 现 ， 构 建 过 程 是 从 基 类 “向 外 ”扩散 的 ， 所 以 基 类 在 导出 类 构造 器 可 以 访问 它 之 
前 ， 就 已 经 完成 了 初始 化 。 即 使 你 不 为 Cartoon0 创 建构 造 器 ， 编 译 器 也 会 为 你 合成 一 个 默认 的 
构造 器 ， 该 构造 器 将 调用 基 类 的 构造 器 。 

练习 3 (2) 证 明 前 面 这 句 话 。 

练习 4: (2) 证 明基 类 构造 器 : (a) 总 是 会 被 调用 ， (b) 在 导出 类 构造 器 之 前 被 调用 。 
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练习 5: (1) 创建 两 个 带 有 默认 构造 器 ( 空 参数 列表 ) 的 类 A 和 类 B。 从 A 中 继承 产生 一 个 名 为 C 
的 新 类 ， 并 在 C 内 创建 一 个 B 类 的 成 员 。 不 要 给 C 编 写 构造 器 。 创 建 一 个 C 类 的 对 象 并 观察 其 结果 。 

带 参数 的 构造 器 

上 例 中 各 个 类 均 含有 默认 的 构造 器 ， 即 这 些 构造 器 都 不 带 参数 。 编 译 器 可 以 轻松 地 调用 它 
们 是 因为 不 必 考虑 要 传递 什么 样 的 参数 的 问题 。 但 是 ， 如 果 没 有 默认 的 基 类 构造 器 ， 或 者 想 调 
用 一 个 带 参数 的 基 类 构造 器 ， 就 必须 用 关键 字 super 显 式 地 编写 调用 基 类 构造 器 的 语句 ， 并 且 配 
以 适当 的 参数 列表 : 

11: reusing/Chess.java 


// Inheritance, constructors and arguments. 
import static net.mindview.util.Print.*; 





class Game { 
Game(int i) { 
print ("Game constructor"); 


$ 
} 


class BoardGame extends Game { 
BoardGame(int i) { 
super (í); 
print("BoardGame constructor"); 
} 
} 


public class Chess extends BoardGame { 
Chess() { 
super (11); 
print("Chess constructor"); 


t 
public static void main(String[] args) { 
Chess x = new Chess(); 
} 
} /* Output: 
Game constructor 
BoardGame constructor 
Chess constructor 
Whim 


如 果 不 在 BoardGame0 中 调用 基 类 构造 器 ， 编 译 器 将 “抱怨 ”无 法 找到 符合 Game0O 形 式 的 
构造 器 。 而 且 ， 调 用 基 类 构造 器 必须 是 你 在 导出 类 构造 器 中 要 做 的 第 一 件 事 (如 果 你 做 错 了 ， 
编译 器 会 提醒 你 )。 

练习 6 (1) 用 Chess.java 来 证 明 前 一 段 话 。 

练习 7，(1) 修改 练习 5， 使 A 和 B 以 带 参 数 的 构造 器 取代 默认 的 构造 器 。 为 C 写 一 个 构造 器 ， 
并 在 其 中 执行 所 有 的 初始 化 。 

练习 8: (1) 创建 一 个 基 类 ， 它 仅 有 一 个 非 默认 构造 器 ， 再 创建 一 个 导出 类 ， 它 带 有 默认 构 
造 器 和 非 默认 构造 器 。 在 导出 类 的 构造 器 中 调用 基 类 的 构造 器 。 

练习 9: (2) 创建 一 个 Root 类 ， 令 其 含有 名 为 Component 1, Component 2, Component 3 的 
类 的 各 一 个 实例 (这些 也 由 你 写 ) 。 从 Root 中 派生 一 个 类 Stem， 也 含有 上 述 各 “组 成 部 分 "。 所 
有 的 类 都 应 带 有 可 打印 出 类 的 相关 信息 的 默认 构造 器 。 

练习 10: (1) 修改 练习 10， 使 每 个 类 都 仅 具 有 非 默 认 的 构造 器 。 


7.3 代理 
第 三 种 关系 称 为 代理 ，Java 并 没有 提供 对 它 的 直接 支持 。 这 是 继承 与 组 合 之 间 的 中 庸 之 道 ， 
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为 我 们 将 一 个 成 员 对 象 置 于 所 要 构造 的 类 中 (就 像 组 合 ) ， 但 与 此 同时 我 们 在 新 类 中 暴露 了 该 
成 员 对 象 的 所 有 方法 〈 就 像 继承 )。 例 如 ， 太 空 船 需要 一 个 控制 模块 : 


17: reusing/SpaceShipControls. java 


public class SpaceShipControls { 
void up(int velocity) {} 
void down(int velocity) {} 
void left(int velocity) {} 
void right(int velocity) {} 
void forward(int velocity) {} 
void back(int velocity) {} 
void turboBoost() {} 

} Mi~ 


构造 太空 船 的 一 种 方式 是 使 用 继承 : 


11: reusing/SpaceShip. java 


public class SpaceShip extends SpaceShipControls { 
private String name; 
public SpaceShip(String name) { this.name = name; } 
public String toString() { return name; } 
public static void main(String[] args) { 
SpaceShip protector = new SpaceShip("NSEA Protector"); 
protector. forward(10@) ; 


} 
} Mi~ 


然而 ，SpaceShip 并 非 真 正 的 SpaceShipControls 类 型 ， 即 便 你 可 以 “告诉 ”SpaceShip 向 前 
运动 (forward())。 更 准确 地 讲 ，SpaceShip 包 含 SpaceShipControls， 与 此 同时 ，SpaceShip- 
Controls 的 所 有 方法 在 SpaceShip 中 都 暴露 了 出 来 。 代 理解 决 了 此 难题 : 


1/: reusing/SpaceShipDelegation. java 


public class SpaceShipDelegation { 
private String name; 
private SpaceShipControls controls = 
new SpaceShipControts() ; 
public SpaceShipDelegation(String name) { 
this.name = name; 


} 
// Delegated methods: 


public void back(int velocity) { 
controls.back(velocity) ; 


$ 
public void down(int velocity) { 
controls.down(velocity) ; 


} 
public void forward(int velocity) { 
controls. forward(velocity); 


} 
public void left(int velocity) { 
controls. left (velocity); 


$ 
public void right(int velocity) { 
controls.right (velocity); 


public void turboBoost() { 
controls. turboBoost () ; 


和 

public void up(int velocity) { 
controls.up(velocity); 

} 

public static void main(String[] args) { 
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SpaceShipDelegation protector = 
new SpaceShipDelegation("NSEA Protector"); 
protector. forward (180) ; 


} 

} H~ 

可 以 看 到 ， 上 面 的 方法 是 如 何 转 递 给 了 底层 的 controls 对 象 ， 而 其 接口 由 此 也 就 与 使 用 继承 
得 到 的 接口 相同 了 。 但 是 ， 我 们 使 用 代理 时 可 以 拥有 更 多 的 控制 力 ， 因 为 我 们 可 以 选择 只 提供 
在 成 员 对 象 中 的 方法 的 某 个 子 集 。 

尽管 Java 语 言 不 直接 支持 代理 ， 但 是 很 多 开发 工具 却 支持 代理 。 例 如 ， 使 用 JetBrains Idea 
IDE 就 可 以 自动 生成 上 面 的 例子 。 

练习 11，(3) 修改 Detergentjava， 让 它 使 用 代理 。 


7.4 结合 使 用 组 合 和 继承 


同时 使 用 组 合 和 继承 是 很 常见 的 事 。 下 例 就 展示 了 同时 使 用 这 两 种 技术 ， 并 配 以 必要 的 构 
造 器 初始 化 ， 来 创建 更 加 复杂 的 类 : 


//: reusing/PlaceSetting. java 
// Combining composition & inheritance. 
import static net.mindview.util.Print.*; 


class Plate { 
Platecint i) { 
print("Plate constructor"); 
} 
$ 


class DinnerPlate extends Plate { 
DinnerPlate(int i) { 
super (i); 
print("DinnerPlate constructor"); 
} 
} 


class Utensil { 
Utensil(int i) { 
print("Utensil constructor"); 
} 


class Spoon extends Utensil { 
Spoon(int 4) { 
super (i); 
print("Spoon constructor"); 
} 
} 


class Fork extends Utensil { 
Fork(int 1) { 
super (i); 
print("Fork constructor”); 
} 
} 


class Knife extends Utensil { 
Knife(int i) { 
super (i); 
print("Knife constructor"); 
} 
} 


// A cultural way of doing something: 
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class Custom { 
Custom(int i) { 
print("Custom constructor"); 
ri 
} 


public class PlaceSetting extends Custom { 
Private Spoon sp; 
private Fork frk; 
private Knife kn; 
private DinnerPlate pl; 
public PlaceSetting(int i) { 
super(i + 1); 
sp = new Spoon(i + 2); 
frk = new Fork(i + 3); 
kn = new Knife(i + 4); 
pl = new DinnerPlate(i + 5); 
print(*PlaceSetting constructor"); 
} 
public static void main(String[] args) { 
PlaceSetting x = new PlaceSetting(9); 


} 

/* Output: 

Custom constructor 
Utensil constructor 
Spoon constructor 
Utensil constructor 
Fork constructor 
Utensil constructor 
Knife constructor 

Plate constructor 
DinnerPlate constructor 
PlaceSetting constructor 
Whim 


虽然 编译 器 强制 你 去 初始 化 基 类 ， 并 且 要 求 你 要 在 构造 器 起 始 处 就 要 这 么 做 ， 但 是 它 并 不 
监督 你 必须 将 成 员 对 象 也 初始 化 ， 因 此 在 这 一 点 上 你 自己 必须 时 刻 注意 。 

这 些 类 如 此 清晰 地 分 离 着 实 使 惊讶。 甚至 不 需要 这 些 方法 的 源 代码 就 可 以 复 用 这 些 代码 ， 
我 们 至 多 只 需要 导 人 一 个 包 。( 对 于 继承 与 组 合 来 说 都 是 如 此 。) 
7.4.1 确保 正确 清理 

Java 中 没有 C++ 中 析 构 函数 的 概念 。 析 构 函数 是 一 种 在 对 象 被 销毁 时 可 以 被 自动 调用 的 函数 。 
其 原因 可 能 是 因为 在 Java 中 ， 我 们 的 习惯 只 是 忘掉 而 不 是 销毁 对 象 ， 并 且 让 垃圾 回收 器 在 必要 
时 释放 其 内 存 。 

通常 这 样 做 是 好 事 ， 但 有 时 类 可 能 要 在 其 生命 周期 内 执行 一 些 必需 的 清理 活动 。 正 如 我 们 
在 第 5 章 中 所 提 到 的 那样 ， 你 并 不 知道 垃圾 回收 器 何 时 将 会 被 调用 ， 或 者 它 是 否 将 被 调用 。 因 此 ， 
如 果 你 想 要 某 个 类 清理 一 些 东 西 ， 就 必须 显 式 地 编写 一 个 特殊 方法 来 做 这 件 事 ， 并 要 确保 客户 
端 程序 员 知晓 他 们 必须 要 调用 这 一 方法 。 就 像 在 第 12 章 所 描述 的 那样 ， 其 首要 任务 就 是 ， 必 须 
将 这 一 清理 动作 置 于 finally 子 句 之 中 ， 以 预防 异常 的 出 现 。 

请 思考 一 下 下 面 这 个 能 在 屏幕 上 绘制 图 案 的 计算 机 辅助 设计 系统 示例 : 


//: reusing/CADSystem. java 
// Ensuring proper cleanup. 

package reusing: 

import static net.mindview.util.Print.*: 


class Shape { 
Shape(int i) { print("Shape constructor"): } 
void dispose() { print("Shape dispose"): } 
} 
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class Circle extends Shape { 
Circle(int i) { 
super (i); 
print ("Drawing Circle"); 
y 
void dispose() { 
print ("Erasing Circle"); 
super .dispose(); 
} 
} 


class Triangle extends Shape { 
Triangle(int i) { 
super (i); 
print("Drawing Triangle"); 
} 
void dispose() { 
print("Erasing Triangle 
super .dispose() ; 
} 
} 





class Line extends Shape { 
private int start, end; 
Line(int start, int end) { 
super (start); 
this.start = start; 
this.end = end; 
print("Drawing Line: " + start +", " + end); 
} 
void dispose() { 
print("Erasing Line: "+ start +", " + end); 
super .dispose(); 
} 
} 


public class CADSystem extends Shape { 
private Circle c; 
private Triangle t; 
private Line[] lines = new Line{3]; 
public CADSystem(int i) { 
super(i + 1); 
for(int j = @; j < Lines.length; j++) 
lines{j] = new Line(j, j*j 
c = new Circle(1); 
t = new Triangle(1); 
print("Combined constructor"): 
} 
public void dispose() { 
print ("CADSystem.dispose()"); 
// The order of cleanup is the reverse 
// of the order of initialization: 
t.dispose(); 
c.dispose( 
for(int i = lines.length - 1; i >= 
Lines {1} .dispose() ; 
super. dispose(); 
} 
public static void main(Stringl] args) { 
CADSystem x = new CADSystem(47) ; 
try { 
// Code and exception handling... 
} finally { 
x.dispose(); 
} 
? 
} /* Output: 
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Shape constructor 

Shape constructor 

Drawing Line: 0, @ 

Shape constructor 

Drawing Line: 1, 1 

Shape constructor 

Drawing Line: 2, 4 

Shape constructor 

Drawing Circle 

Shape constructor 

Drawing Triangle 

Combined constructor 

CADSystem.dispose() 

Erasing Triangle 

Shape dispose 

Erasing Circle 

Shape dispose 

Erasing Line: 2, 4 

Shape dispose 

Erasing Line: 1, 1 

Shape dispose 

Erasing Line: 0, @ 

Shape dispose 

Shape dispose 

Winx 

此 系统 中 的 一 切 都 是 某 种 Shape (Shape 自 身 就 是 一 种 Object, 因为 Shape 继 承 自 根 类 Object) 。 
每 个 类 都 禾 写 Shape 的 dispose0 方 法 ， 并 运用 super 来 调用 该 方法 的 基 类 版 本 。 尽 管 对 象 生命 期 
中 任何 被 调用 的 方法 都 可 以 做 一 些 必需 的 清理 工作 ， 但 是 Circle、Triangle 和 Line 这 些 特定 的 
Shape 类 仍然 都 带 有 可 以 进行 “绘制 ”的 构造 器 。 每 个 类 都 有 自己 的 dispose() 方 法 将 未 存 于 内 存 
之 中 的 东西 恢复 到 对 象 存在 之 前 的 状态 。 

在 main0 中 可 以 看 到 try 和 finally 这 两 个 之 前 还 没有 看 到 过 的 关键 字 ， 我 们 将 在 第 12 章 对 它 
们 进行 详细 解释 。 关 键 字 try 表 示 ， 下 面 的 块 (用 一 组 大 括号 括 起 来 的 范围 ) 是 所 谓 的 保护 区 
(guarded region) ,这 意味 着 它 需要 被 特殊 处 理 。 其 中 一 项 特殊 处 理 就 是 无 论 try 块 是 怎样 退出 的 ， 
保护 区 后 的 finally 子 句 中 的 代码 总 是 要 被 执行 的 。 这 里 finally 子 句 表示 的 是 “无 论 发 生 什 么 事 ， 
一 定 要 为 x 调用 dispose0”。 

在 清理 方法 (dispose0) H, 还 必须 注意 对 基 类 清理 方法 和 成 员 对 象 清理 方法 的 调用 顺序 ， 
以 防 某 个 子 对 象 依赖 于 另 一 个 子 对 象 情形 的 发 生 。 一 般 而 言 ， 所 采用 的 形式 应 该 与 C++ 编译 器 
在 其 析 构 函数 上 所 施加 的 形式 相同 : 首先 ， 执 行 类 的 所 有 特定 的 清理 动作 ， 其 顺序 同 生成 顺 
序 相反 (通常 这 就 要 求 基 类 元 素 仍旧 存活 ) ， 然 后 ， 就 如 我 们 所 示范 的 那样 ， 调 用 基 类 的 清 
理 方法 。 

许多 情况 下 ， 清 理 并 不 是 问题 ， 仅 需 让 垃圾 回收 器 完成 该 动作 就 行 。 但 当 必须 亲自 处 理 清 
理 时 ， 就 得 多 做 努力 并 多 加 小 心 。 因为， 一旦 涉及 垃圾 回收 ， 能 够 信赖 的 事 就 不 会 很 多 了 。 垃 
圾 回收 器 可 能 永远 也 无 法 被 调用 ， 即 使 被 调用 ， 它 也 可 能 以 任何 它 想 要 的 顺序 来 回收 对 象 。 最 
好 的 办 法 是 除了 内 存 以 外 ， 不 能 依赖 垃圾 回收 器 去 做 任何 事 。 如 果 需 要 进行 清理 ， 最 好 是 编写 
你 自己 的 清理 方法 ， 但 不 要 使 用 finalize0 。 

练习 12: (3) 将 一 个 适当 的 dispose0 方 法 的 层次 结构 添加 到 练习 9 的 所 有 类 中 。 
7.4.2 名 称 屏蔽 

如 果 Java 的 基 类 拥有 某 个 已 被 多 次 重 载 的 方法 名 称 ， 那 么 在 导出 类 中 重新 定义 该 方法 名 称 
并 不 会 屏蔽 其 在 基 类 中 的 任何 版 本 (这 一 点 与 C++ 不 同 )。 因 此 ， 无 论 是 在 该 层 或 者 它 的 基 类 中 
对 方法 进行 定义 ， 重 载 机 制 都 可 以 正常 工作 : 
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11: reusing/Hide. java 
// Overloading a base-class method name in a derived 
// class does not hide the base-class versions. 
import static net.mindview.util.Print.*; 


class Homer { 
char doh(char c) { 
print("doh(char)"); 
return 'd'; 


} 
float doh(float f) { 
print ("doh(float)"); 
return 1.0f; 
$ 
} 


class Milhouse {} 
class Bart extends Homer { 
void doh(Milhouse m) { 
print("doh(Milhouse)"); 


} 


public class Hide { 
public static void main(String[] args) { 
Bart b = new Bart(); 





b.don(1.0f); 
bidoh(new Mithouse()): 


} 
} /* Output: 
doh(float) 
doh(char) 
doh(float) 
doh(Hilhouse) 
Whim 


可 以 看 到 ， 虽 然 Bart31 和 人 了 一 个 新 的 重 载 方法 〈 在 C++ 中 若 要 完成 这 项 工作 则 需要 屏蔽 基 类 
方法 )， 但 是 在 Bart 中 Homer 的 所 有 重 载 方法 都 是 可 用 的 。 正 如 读者 将 在 下 一 章 所 看 到 的 ， 使 用 
与 基 类 完全 相同 的 特征 签名 及 返回 类 型 来 覆盖 具有 相同 名 称 的 方法 ， 是 一 件 极其 平常 的 事 。 但 
它 也 令 人 迷惑 不 解 〈 这 也 就 是 为 什么 C++ 不 允许 这 样 做 的 原因 所 在 -防止 你 可 能 会 犯错 误 ) 。 

Java SE5 新 增加 了 @Override 注 解 ， 它 并 不 是 关键 字 ， 但 是 可 以 把 它 当 作 关 键 字 使 用 。 当 你 
想 要 种 写 某 个 方法 时 ， 可 以 选择 添加 这 个 注解 ， 在 你 不 留心 重 载 而 并 非 覆 写 了 该 方法 时 ， 编 译 
器 就 会 生成 一 条 错误 消息 : 


/1/: reusing/Lisa.java 
// {CompileTimeError} (Won't compile) 


class Lisa extends Homer { 
@Override void doh(Milhouse m) { 
System. out .printin("doh(MiLhouse)") ; 


} 
dh 
{CompileTimeError} 标 签 把 该 文件 从 本 书 的 Ant 构 建 中 排除 了 出 来 ， 但 是 如 果 手 工 编译 该 文 
件 ， 就 会 看 到 下 面 的 错误 消息 : 
method does not override a method from its superclass 
这 样 ，@Override 注 解 可 以 防止 你 在 不 想 重 载 时 而 意外 地 进行 了 重 载 。 
练习 13: (2) 创建 一 个 类 ， 它 应 带 有 一 个 被 重 载 了 三 次 的 方法 。 继 承 产生 一 个 新 类 ， 并 添加 
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一 个 该 方法 的 新 的 重 载 定义 ， 展 示 这 四 个 方法 在 导出 类 中 都 是 可 以 使 用 的 。 
7.5 在 组 合 与 继承 之 间 选 择 


组 合 和 继承 都 允许 在 新 的 类 中 放置 子 对 象 ， 组 合 是 显 式 地 这 样 做 ， 而 继承 则 是 隐 式 地 做 。 
读者 或 许 想 知道 二 者 间 的 区 别 何在 ， 以 及 怎样 在 二 者 之 间 做 出 选择 。 

组 合 技术 通常 用 于 想 在 新 类 中 使 用 现 有 类 的 功能 而 非 它 的 接口 这 种 情形 。 即 ， 在 新 类 中 供 
人 和 某 个 对 象 ， 让 其 实现 所 需要 的 功能 ， 但 新 类 的 用 户 看 到 的 只 是 为 新 类 所 定义 的 接口 ， 而 非 所 
移入 对 象 的 接口 。 为 取得 此 效果 ， 需 要 在 新 类 中 伐 人 一 个 现 有 类 的 private 对 象 。 

有 了 时， 允许 类 的 用 户 直接 访问 新 类 中 的 组 合成 分 是 极 具 意义 的 ， 也 就 是 说 ， 将 成 员 对 象 声 
明 为 public。 如 果 成 员 对 象 自身 都 隐藏 了 具体 实现 ， 那 么 这 种 做 法 是 安全 的 。 当 用 户 能 够 了 解 到 
你 正在 组 装 一 组 部 件 时 ， 会 使 得 端口 更 加 易于 理解 。car 对 象 即 为 一 个 很 好 的 例子 : 


/1/: reusing/Car.java 
// Composition with public objects. 


class Engine { 
public void start() {} 
public void rev() {} 
public void stop() {} 
} 


class Wheel { 
public void inflate(int psi) {} 


class Window { 
public void rollup() {} 
public void rolldown() {} 
} 


class Door { 
public Window window = new Window(); 
public void open() {} 
public void close() {} 

} 


public class Car { 

public Engine engine = new Engine(); 
public Wheel[] wheel = new Wheel [4] ; 
public Door 

left = new Door(), 

right = new Door(); // 2-door 
public Car() { 

for(int 1 = 0; i < 4: i++) 

wheel[i] = new Wheel(); 


} 

public static void main(String[] args) { 
Car car = new Car(); 
car.left.window.rollup(); 
car.wheel [0]. inflate(72); 


) 
} H~ 
由 于 在 这 个 例子 中 ear 的 组 合 也 是 问题 分 析 的 一 部 分 (而 不 仅仅 是 底层 设计 的 一 部 分 )， 所 
以 使 成 员 成 为 pubiic 将 有 助 于 客户 端 程序 员 了 解 怎样 去 使 用 类 ， 而 且 也 降低 了 类 开发 者 所 面临 的 
代码 复杂 度 。 但 务必 要 记得 这 仅仅 是 一 个 特例 ， 一 般 情 况 下 应 该 使 域 成 为 private。 
在 继承 的 时 候 ， 使 用 某 个 现 有 类 ， 并 开发 一 个 它 的 特殊 版 本 。 通 常 ， 这 意味 着 你 在 使 用 一 
个 通用 类 ， 并 为 了 某 种 特殊 需要 而 将 其 特殊 化 。 略 微 思 考 一 下 就 会 发 现 ， 用 一 个 “交通 工具 ” 
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对 象 来 构成 一 部 “车 子 ”是 毫 无 意义 的 ， 因 为 “车 子 ” 并 不 包含 “交通 工具 ”"， 它 仅 是 一 种 交通 
工具 (is-a 关 系 )。“is-a”( 是 一 个 ) 的 关系 是 用 继承 来 表达 的 ， 而 “has-a”( 有 一 个 ) 的 关系 则 
是 用 组 合 来 表达 的 。 

练习 14; (1) 在 Carjava 中 给 Engine 添 加 一 个 service0 方 法 ， 并 在 main0 中 调用 该 方法 。 


7.6 protected 关 键 字 


现在 ， 我 们 已 介绍 完了 继承 ， 关 键 字 proteeted 最 终 具 有 了 意义 。 在 理想 世界 中 ， 仅 靠 关 键 
字 private 就 已 经 足够 了 。 但 在 实际 项 目 中 ， 经 常会 想 要 将 某 些 事物 尽 可 能 对 这 个 世界 隐藏 起 来 ， 
但 仍然 允许 导出 类 的 成 员 访问 它们 。 

关键 字 protected 就 是 起 这 个 作用 的 。 它 指明 “就 类 用 户 而 言 ， 这 是 private 的 ， 但 对 于 任何 
继承 于 此 类 的 导出 类 或 其 他 任何 位 于 同一 个 包 内 的 类 来 说 ， 它 却 是 可 以 访问 的 。” (protected 也 
提供 了 包 内 访问 权限 。) 

尽管 可 以 创建 protected 域 ， 但 是 最 好 的 方式 还 是 将 域 保持 为 private， 你 应 当 一 直 保留 “更 
改 底层 实现 ”的 权利 。 然 后 通过 protected 方 法 来 控制 类 的 继承 者 的 访问 权限 。 


44: reusing/0rc.java 
// The protected keyword. 
import static net.mindview.util.Print.*; 


class Villain { 
private String name; 
Protected void set(String nm) { name = nm; } 
public Villain(String name) { this.name = name; } 
public String toString() { 

return "I'm a Villain and my name is " + name; 

1 

} 


public class Orc extends Villain { 
private int orcNumber; 
public Orc(String name, int orcNumber) { 
super (name) ; 
this.orcNumber = orcNumber; 


Public void change(String name, int orcNumber) { 
set(name); // Available because it's protected 
this.orcNumber = orcNumber; 


t 
public String toString() { 
return "Orc " + orcNumber + ": “ + super.toString(): 


} 

public static void main(String{] args) { 
Orc orc = new Orc("Limburger”, 12); 
print(orc): 
orc.change("Bob", 19); 
print(orc); 


$ 
} /* Output: 
Orc 12: I'm a Villain and my name is Limburger 
Orc 19: I'm a Villain and my name is Bob 
i~ 


可 以 发 现 ，change0 可 以 访问 set0， 这 是 因为 它 是 protected 的 。 还 应 注意 Ore 的 toString0 方 
法 的 定义 方式 ， 它 依据 toString0 的 基 类 版 本 而 定义 。 

练习 15: (2) 在 包 中 编写 一 个 类 ， 类 应 具备 一 个 protected 方 法 。 在 包 外 部 ， 试 着 调用 访 
Protected 方 法 并 解释 其 结果 。 然 后 ， 从 你 的 类 中 继承 产生 一 个 类 ， 并 从 该 导出 类 的 方法 内 部 调 
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用 该 protected 方 法 。 
7.7 向 上 转型 


“为 新 的 类 提供 方法 ”并 不 是 继承 技术 中 最 重要 的 方面 ， 其 最 重要 的 方面 是 用 来 表现 新 类 和 
基 类 之 间 的 关系 。 这 种 关系 可 以 用 “新 类 是 现 有 类 的 一 种 类 型 ”这 句 话 加 以 概括 。 

这 个 描述 并 非 只 是 一 种 解释 继承 的 华丽 的 方式 ， 这 直接 是 由 语言 所 支撑 的 。 例 如 ， 假 设 有 
一 个 称 为 Instrument 的 代表 乐器 的 基 类 和 一 个 称 为 Wind 的 导出 类 。 由 于 继承 可 以 确保 基 类 中 所 
有 的 方法 在 导出 类 中 也 同样 有 效 ， 所 以 能 够 向 基 类 发 送 的 所 有 信息 同样 也 可 以 向 导出 类 发 送 。 
如 果 Instrument 类 具有 一 个 play0 方 法 ， 那 么 Wind 乐 器 也 将 同样 具备 。 这 意味 着 我 们 可 以 准确 
地 说 Wind 对 象 也 是 一 种 类 型 的 Instrument。 下 面 这 个 例子 说 明了 编译 器 是 怎样 支持 这 一 概念 的 ; 


/1/: reusing/Wind. java 
// Inheritance & upcasting. 


class Instrument { 
public, void play() {} 
static void tune(Instrument i) { 
WM vee 
1.playO): 


} 


// Wind objects are instruments 
// because they have the same interface: 
public class Wind extends Instrument { 
public static void main(String{] args) { 
Wind flute = new Wind(); 
Instrument. tune(flute); // Upcasting 


) 

} Mi~ 

在 此 例 中 ，tune0 方 法 可 以 接受 Instruament 引用 ， 这 实在 太 有 趣 了 。 但 在 Wind.main() 中 ， 
传递 给 tune0 方 法 的 是 一 个 Wind 引用 。 鉴 于 Java 对 类 型 的 检查 十 分 严格 ， 接 受 某 种 类 型 的 方法 
同样 可 以 接受 另外 一 种 类 型 就 会 显得 很 奇怪 ， 除 非 你 认识 到 Wind 对 象 同样 也 是 一 种 Instrument 
对 象 ， 而 且 也 不 存在 任何 tune0 方 法 是 可 以 通过 Instrument 来 调用 ， 同 时 又 不 存在 于 Wind 之 中 。 
在 tune0 中 ， 程 序 代码 可 以 对 Instrument 和 它 所 有 的 导出 类 起 作用 ， 这 种 将 Wind 引 用 转换 为 
Instrument 引用 的 动作 ， 我 们 称 之 为 向 上 转型 。 


7.7.1 为 什么 称 为 向 上 转型 

该 术语 的 使 用 有 其 历史 原因 ， 并 且 是 以 传统 的 类 继承 图 的 绘制 方法 为 基础 的 ;将 根 置 于 页 
面 的 顶端 ， 然 后 逐渐 向 下 。 (当然 也 可 以 以 任何 你 认为 有 效 的 方法 进行 绘制 。) FE, Wind java 
的 继承 图 就 是 〈 如 右 图 所 示 ) : D 

由 导出 类 转型 成 基 类 ， 在 继承 图 上 是 向 上 移动 的 ， 因 此 一 般 称 为 向 上 转 。 | Instrument 
型 。 由 于 向 上 转型 是 从 一 个 较 专用 类 型 向 较 通用 类 型 转换 ， 所 以 总 是 很 安全 人 
的 。 也 就 是 说 ， 导 出 类 是 基 类 的 一 个 超 集 。 它 可 能 比 基 类 含有 更 多 的 方法 ， 
但 它 必须 至 少 具备 基 类 中 所 含有 的 方法 。 在 向 上 转型 的 过 程 中 ， 类 接口 中 唯 
一 可 能 发 生 的 事情 是 丢失 方法 ， 而 不 是 获取 它们 。 这 就 是 为 什么 编译 器 在 “未 曾 明确 表示 转型 ” 
或 “未 曾 指定 特殊 标记 ”的 情况 下 ,仍然 允许 向 上 转型 的 原因 。 

也 可 以 执行 与 向 上 转型 相反 的 向 下 转型 ， 但 其 中 含有 一 个 难题 ， 这 将 在 第 8 章 和 第 14 章 中 进 
一 步 解释 。 
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7.7.2 再 论 组 合 与 继承 

在 面向 对 象 编程 中 ， 生 成 和 使 用 程序 代码 最 有 可 能 采用 的 方法 就 是 直接 将 数据 和 方法 包装 
进 一 个 类 中 ， 并 使 用 该 类 的 对 象 。 也 可 以 运用 组 合 技术 使 用 现 有 类 来 开发 新 的 类 ， 而 继承 技术 
其 实 是 不 太 常用 的 。 因 此 ， 尽 管 在 教授 OOP 的 过 程 中 我 们 多 次 强调 继承 ， 但 这 并 不 意味 着 要 尽 
可 能 使 用 它 。 相 反 ， 应 当 慎 用 这 一 技术 ， 其 使 用 场合 仅 限 于 你 确信 使 用 该 技术 确实 有 效 的 情况 。 
到 底 是 该 用 组 合 还 是 用 继承 ， 一 个 最 清晰 的 判断 办 法 就 是 问 一 问 自己 是 否 需要 从 新 类 向 基 类 进 
行 向 上 转型 。 如 果 必 须 向 上 转型 ， 则 继承 是 必要 的 ， 但 如 果 不 需 要 ， 则 应 当 好 好 考虑 自己 是 否 
需要 继承 。 第 8 章 提出 了 一 个 使 用 向 上 转型 的 最 具 说 服 力 的 理由 ， 但 只 要 记得 自问 一 下 “我 真 的 
需要 向 上 转型 吗 ? ”就 能 较 好 地 在 这 两 种 技术 中 做 出 决定 。 

练习 16; (2) 创建 一 个 名 为 Amphibian 的 类 。 由 此 继承 产生 一 个 称 为 Frog 的 类 。 在 基 类 中 设 
置 适当 的 方法 。 在 main0 中 ， 创 建 一 个 Frog 并 向 上 转型 至 Amphibian， 然 后 说 明 所 有 方法 都 可 
工作 。 

练习 17: (1) 修改 练习 16， 使 Frog 覆 盖 基 类 中 方法 的 定义 (QR SCR eet 
名 )。 请 留心 main0 中 都 发 生 了 什么 。 


7.8 final 关 键 字 


根据 上 下 文 环境 ，Java 的 关键 字 final 的 含义 存在 着 细微 的 区 别 ， 但 通常 它 指 的 是 “这 是 无 法 
改变 的 。” 不 想 做 改变 可 能 出 于 两 种 理由 : 设计 或 效率 。 由 于 这 两 个 原因 相差 很 远 ， 所 以 关键 字 
final 有 可 能 被 误 用 。 

以 下 几 节 谈论 了 可 能 使 用 到 final 的 三 种 情况 : 数据 、 方 法 和 类 。 

7.8.1 final 数据 

许多 编程 语言 都 有 某 种 方法 ， 来 向 编译 器 告知 一 块 数据 是 恒定 不 变 的 。 有 时 数据 的 恒定 不 
变 是 很 有 用 的 ， 比 如 : 

1. 一 个 永 不 改变 的 编译 时 常量 。 

2. 一 个 在 运行 时 被 初始 化 的 值 ， 而 你 不 希望 它 被 改变 。 

对 于 编译 期 常量 这 种 情况 ， 编 译 器 可 以 将 该 常量 值 代入 任何 可 能 用 到 它 的 计算 式 中 ， 也 就 
是 说 ， 可 以 在 编译 时 执行 计算 式 ， 这 减轻 了 一 些 运行 时 的 负担 。 在 Java 中 ， 这 类 常量 必须 是 基 
本 数据 类 型 ， 并 且 以 关键 字 final 表 示 。 在 对 这 个 常量 进行 定义 的 时 候 ， 必 须 对 其 进行 赋值 。 

一 个 既是 static 又 是 final 的 域 只 占据 一 段 不 能 改变 的 存储 空间 。 

当 对 对 象 引 用 而 不 是 基本 类 型 运用 final 时 ， 其 含义 会 有 一 点 令 人 迷惑 。 对 于 基本 类 型 ， 
final 使 数值 恒定 不 变 ， 而 用 于 对 象 引 用 ，final 使 引用 恒定 不 变 。 一 旦 引用 被 初始 化 指向 一 个 对 
象 ， 就 无 法 再 把 它 改 为 指向 另 一 个 对 象 。 然 而 ， 对 象 其 自身 却 是 可 以 被 修改 的 ，Java 并 未 提供 
使 任何 对 象 恒定 不 变 的 途径 (但 可 以 自己 编写 类 以 取得 使 对 象 恒 定 不 变 的 效果 )。 这 一 限制 同样 
适用 数组 ， 它 也 是 对 象 。 

下 面 的 示例 示范 了 final 域 的 情况 。 注 意 ， 根 据 惯例 ， 既 是 static 又 是 final 的 域 ( 即 编译 期 党 
it) 将 用 大 写 表示 ， 并 使 用 下 划 线 分 隔 各 个 单词: 


//:_reusing/FinalData. java 

// The effect of final on fields. 
import java.util.*; 

import static net.mindview.util.Print.*; 


class Value { 
int 4; // Package access 
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public Value(int i) { this.i = ii } 


public class FinalData { 
private static Random rand = new Random(47); 
private String id; 
public FinalData(String id) { this.id =. id; } 
// Can be compile-time constants: 
private final int valueOne = 9: 
private static final int VALUE_TWO = 99; 
// Typical public constant: 
public static final int VALUE_THREE = 39; 
// Cannot be compile-time constants: 
private final int i4 = rand.nextInt(26); 
static final int INT_5 = rand.nextInt(20); 
private Value v1 = new Value(11);. 
private final Value v2 = new Value(22); 
private static final Value VAL_3 = new Value(33); 
V1 Arrays: 
private final int{] a= { 1, 2,3, 4, 5, 6); 
Public String toString() { 
return id +": " + "i=" +144", INTS =" + INT_5; 


} 

public static void main(String{] args) { 
FinalData fdl = new FinalData("fd1"); 
J/! fdl.valueOne++; // Error; can't change value 
fdl.v2.i++; // Object isn't constant! 
fdl.v1 = new Value(9); // OK -- not final 
for(int i = 0; i < fdl.a.length; i++) 

fdl.ali]++; // Object isn't constant! 

JJ) fd1.v2 = new Value(®); // Error: Can't 
//! {fd1.VAL_3 = new Value(1); // change reference. 
/ML fdl.a = new int {3}; 
print (fd1); 
print("Creating new FinalData"); 
FinalData fd2 = new FinalData("fd2"); 
print(fd1); 
print(fd2); 


} h Output: 

fdl: i4 = 15, INT_S = 18 

Creating new FinalData 

fdl: 14 = 15, INT_S = 18 

fd2: 14 = 13, INT-5 = 18 

Whim 

由 于 valuOne 和 VAL_TWO 都 是 带 有 编译 时 数值 的 final 基 本 类 型 ， 所 以 它们 二 者 均 可 以 用 作 
编译 期 常量 ， 并 且 没有 重大 区 别 。VAL_THREE 是 一 种 更 加 典型 的 对 常量 进行 定义 的 方式 : E 
义 为 public， 则 可 以 被 用 于 包 之 外 ， 定 义 为 static， 则 强调 只 有 一 份 ， 定 义 为 final， 则 说 明 它 是 一 
个 常量 。 请 注意 ， 带 有 恒定 初始 值 ( 即 ， 编 译 期 常量 ) 的 final static 基 本 类 型 全 用 大 写字 母 命名 ， 
并 且 字 与 字 之 间 用 下 划 线 隔 开 (这 就 像 C 常 量 一 样 ，C 常 量 是 这 一 命名 传统 的 发 源 地 ) 。 

我 们 不 能 因为 某 数据 是 final 的 就 认为 在 编译 时 可 以 知道 它 的 值 。 在 运行 时 使 用 随机 生成 的 
数值 来 初始 化 4 和 INT_5 就 说 明了 这 一 点 。 示 例 部 分 也 展示 了 将 final 数 值 定义 为 静态 和 非 静态 的 
区 别 。 此 区 别 只 有 当 数 值 在 运行 时 内 被 初始 化 时 才 会 显现 ， 这 是 因为 编译 器 对 编译 时 数值 一 视 
同仁 《并 且 它们 可 能 因 优化 而 消失 )。 当 运行 程序 时 就 会 看 到 这 个 区 别 。 请 注意 ， 在 fal 和 fd2 中 ， 
这 的 值 是 唯一 的 ， 但 INT_5 的 值 是 不 可 以 通过 创建 第 二 个 FinaIData 对 象 而 加 以 改变 的 。 这 是 因 
为 它 是 static 的 ， 在 装载 时 已 被 初始 化 ， 而 不 是 每 次 创建 新 对 象 时 都 初始 化 。 

V1 到 VAL_3 这 些 变量 说 明了 final 引 用 的 意义 。 正 如 在 main0 中 所 看 到 的 ， 不 能 因为 v2 是 final 
的 ， 就 认为 无 法 改变 它 的 值 。 由 于 它 是 一 个 引用 ，final 意 味 着 无 法 将 2 再 次 指向 另 一 个 新 的 对 
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象 。 这 对 数组 具有 同样 的 意义 ， 数 组 只 不 过 是 另 一 种 引用 (我 还 不 知道 有 什么 办 法 能 使 数组 引 
用 本 身 成 为 final)。 看 起 来 ， 使 引用 成 为 final 没 有 使 基本 类 型 成 为 final 的 用 处 大 。 

练习 18: (2) 创建 一 个 含有 static final 域 和 final 域 的 类 ， 说 明 二 者 间 的 区 别 。 

空白 final 

Java 允 许 生成 “空白 final" ， 所 谓 空白 final 是 指 被 声明 为 final 但 又 未 给 定 初 值 的 域 。 无 论 什 
么 情况 ， 编 译 器 都 确保 空白 final 在 使 用 前 必须 被 初始 化 。 但 是 ， 空 白 final 在 关键 字 final 的 使 用 
上 提供 了 更 大 的 灵活 性 ， 为 此 ， 一 个 类 中 的 final 域 就 可 以 做 到 根据 对 象 而 有 所 不 同 ， 却 又 保持 
其 恒定 不 变 的 特性 。 下 面 即 为 一 例 ; 


//: reusing/BlankFinal. java 
// "Blank" final fields. 


class Poppet { 
private int i; 
Poppet(int ii) { 1 = i; } 
} 


public class BlankFinal { 

private final int i = @; // Initialized final 
private final int j; // Blank final 
private final Poppet p; // Blank final reference 
// Blank finals MUST be initialized in the constructor: 
public BlankFinal() { 

j= 1; // Initialize blank final 

p = new Poppet(1); // Initialize blank final reference 
} 
public BlankFinal(int x) { 

j = x; // Initialize blank final 

p = new Poppet(x); // Initialize blank final reference 
} 
public static void main(String{] args) { 

new BlankFinal(); 

new BlankF inal (47); 


} 
} Mi~ 


必须 在 域 的 定义 处 或 者 每 个 构造 器 中 用 表达 式 对 final 进 行 赋值 ， 这 正 是 final 域 在 使 用 前 总 
是 被 初始 化 的 原因 所 在 。 

练习 19: (2) 创建 一 个 含有 指向 某 对 象 的 空白 final 引 用 的 类 。 在 所 有 构造 器 内 部 都 执行 空白 
final 的 初始 化 动作 。 说 明 Java 确 保 final 在 使 用 前 必须 被 初始 化 ， 且 一 旦 被 初始 化 即 无 法 改变 。 

final 参数 

Java 允 许 在 参数 列表 中 以 声明 的 方式 将 参数 指明 为 final。 这 意味 着 你 无 法 在 方法 中 更 改 参数 
引用 所 指向 的 对 象 : 

/1/: reusing/FinatArguments.java 

// Using "final" with method arguments. 


class Gizmo { 
public void spin() {} 
} 


public class FinalArguments { 
void with(final Gizmo g) { 
//! g = new Gizmo(); // Ilegal -- g is final 
} 
void without (Gizmo g) { 
g = new Gizmo(); // OK -- g not final 
B.spin(); 
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// void f(final int i) { i++; } // Can't change 
// You can only read from a final primitive: 


int g(final int i) { return i + 1; } 
public static void main(String[} args) { 
FinalArguments bf = new FinalArguments(); 
bf.without (null); 
bf.with(null); 


} 
yh 
方法 f0 和 gO 展示 了 当 基 本 类 型 的 参数 被 指明 为 final 时 所 出 现 的 结果 : 你 可 以 读 参数 ， 但 却 
无 法 修改 参数 。 这 一 特性 主要 用 来 向 匿名 内 部 类 传递 数据 ， 我 们 将 在 第 10 章 中 学 习 它 。 


7.8.2 final 方 法 x 

使 用 final 方 法 的 原因 有 两 个 。 第 一 个 原因 是 把 方法 锁定 ， 以 防 任何 继承 类 修改 它 的 含义 。 
这 是 出 于 设计 的 考虑 ; 想 要 确保 在 继承 中 使 方法 行为 保持 不 变 ， 并 且 不 会 被 覆盖 。 

过 去 建议 使 用 final 方 法 的 第 二 个 原因 是 效率 。 在 Java 的 早期 实现 中 ， 如 果 将 一 个 方法 指明 为 
final， 就 是 同意 编译 器 将 针对 该 方法 的 所 有 调用 都 转 为 内 嵌 调 用 。 当 编译 器 发 现 一 个 final 方 法 
调用 命令 时 ， 它 会 根据 自己 的 谨慎 判断 ， 跳 过 插入 程序 代码 这 种 正常 方式 而 执行 方法 调用 机 制 
(将 参数 压 人 栈 ， 跳 至 方法 代码 处 并 执行 ， 然 后 跳 回 并 清理 栈 中 的 参数 ， 处 理 返 回 值 )， 并 且 以 
方法 体 中 的 实际 代码 的 副本 来 替代 方法 调用 。 这 将 消除 方法 调用 的 开销 。 当 然 ， 如 果 一 个 方法 
很 大 ， 你 的 程序 代码 就 会 膨胀 ， 因 而 可 能 看 不 到 内 做 带 来 的 任何 性 能 提高 ， 因 为 ， 所 带 来 的 性 
能 提高 会 因为 花费 于 方法 内 的 时 间 量 而 被 缩减 。 

在 最 近 的 Java 版 本 中 ， 虚 拟 机 (特别 是 hotspot 技 术 ) 可 以 探测 到 这 些 情况 ， 并 优化 去 掉 这 些 
效率 反而 降低 的 额外 的 内 幅 调 用 ， 因 此 不 再 需要 使 用 final 方 法 来 进行 优化 了 。 事 实 上 ， 这 种 做 
法 正在 逐渐 地 受到 劝阻 。 在 使 用 Java SE5/6 时 ， 应 该 让 编译 器 和 JVM 去 处 理 效 率 问题 ， 只 有 在 想 
要 明确 禁止 覆盖 时 ， 才 将 方法 设置 为 final 的 。 。 

final 和 private 关 键 字 

类 中 所 有 的 private 方 法 都 隐 式 地 指定 为 是 final 的 。 由 于 无 法 取 用 private 方 法 ， 所 以 也 就 无 
法 覆盖 它 。 可 以 对 private 方 法 添加 final 修饰 词 ， 但 这 并 不 能 给 该 方法 增加 任何 额外 的 意义 。 

这 一 问题 会 造成 混 清 。 因 为 ， 如 果 你 试图 覆盖 一 个 private 方 法 〈 隐 含 是 final 的 ) ， 似 乎 是 奏 
效 的 ， 而 且 编译 器 也 不 会 给 出 错误 信息 : 


/1/: reusing/FinalOverridingIllusion.java 
// It only looks like you can override 
// a private or private final method. 
import static net.mindview.util.Print.*; 


class WithFinals { 
// Identical to "private" alone: 
private final void f() { print("WithFinals.f()"); } 
// Also automatically "final": 
private void g() { print("WithFinals.g()"); } 


class OverridingPrivate extends WithFinals { 
private final void f() { 
print("OverridingPrivate.f()"): 
} 


O 不 要 陷入 对 仓促 优化 的 强烈 渴望 之 中 。 如 果 你 的 系统 得 以 运行 ， 而 其 速度 很 慢 ， 使 用 final 关 键 字 来 修复 该 问 
题 是 难以 奏效 的 。 在 http://MindView.neVBooks/BetterJava 可 以 找到 有 关 性 能 测试 的 信息 ， 它 们 有 助 于 提高 你 的 
程序 的 运行 速度 。 
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private void g() { 
print(*OverridingPrivate.g()"); 


È 


class OverridingPrivate2 extends OverridingPrivate { 
public final void fO { 
print("OverridingPrivate2.f()"): 
} 
public void g() { 
print (“OverridingPrivate2.g()"): ` 
} 
p? 
public class FinalOverridingIllusion { 
public static void main(String{] args) { 
OverridingPrivate2 op2 = new OverridingPrivate2(); 
op2.f(); 
op2.80); 
// You can upcast: 
OverridingPrivate op = op2; 
// But you can't call the methods: 
i op.f(); 
I op-BO% 
// Same here: 
WithFinals wf = op2; 
TY wf.f(); 
WL weg: 


} 
} /* Output: 
OverridingPrivate2.f() 
OverridingPrivate2.g() 
Whim 


“BEE” RAERD ARERR BARH. BN, BARES Rel bee 
为 它 的 基本 类 型 并 调用 相同 的 方法 (这 一 点 在 下 一 章 亲 明 )。 如 果 某 方法 为 private， 它 就 不 是 基 
类 的 接口 的 一 部 分 。 它 仅 是 一 些 隐藏 于 类 中 的 程序 代码 ， 只 不 过 是 具有 相同 的 名 称 而 已 。 但 如 
果 在 导出 类 中 以 相同 的 名 称 生成 一 个 public、protected 或 包 访问 权限 方法 的 话 ， 该 方法 就 不 会 产 
生 在 基 类 中 出 现 的 “ 仅 具有 相同 名 称 ” 的 情况 。 此 时 你 并 没有 覆盖 该 方法 ， 仅 是 生成 了 一 个 新 
的 方法 。 由 于 private 方 法 无 法 触及 而 且 能 有 效 隐 藏 ， 所 以 除了 把 它 看 成 是 因为 它 所 归属 的 类 的 
组 织 结构 的 原因 而 存在 外 ， 其 他 任何 事物 都 不 需要 考虑 到 它 。 

练习 20: (1) 展示 @Override 注 解 可 以 解决 本 节 中 的 问题 。 

练习 21: (1) 创建 一 个 带 final 方 法 的 类 。 由 此 继承 产生 一 个 类 并 尝试 覆盖 该 方法 。 
7.8.3 final 类 

当 将 某 个 类 的 整体 定义 为 final 时 (通过 将 关键 字 final 置 于 它 的 定义 之 前 ) ， 就 表明 了 你 不 打 
算 继承 该 类 ， 而 且 也 不 允许 别人 这 样 做 。 换 名 话说， 出 于 某 种 考虑 ， 你 对 该 类 的 设计 永 不 需要 
做 任何 变动 ， 或 者 出 于 安全 的 考虑 ， 你 不 希望 它 有 子 类 。 


//: reusing/Jurasstc.java 
// Making an entire class final. 


class SmaliBrain {} 


final class Dinosaur { 
int 1 = 7; 
int j = 1; 
SmallBrain x = new Smal1Brain(): 
void fO {} 
} 
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/1/! class Further extends Dinosaur {} 
/1/ error: Cannot extend final class ‘Dinosaur’ 


public class Jurassic { 
public static void main(String[] args) { 
Dinosaur n = new Dinosaur(): 
n.f0; 
n.i = 40; 
nitty 
} 

} i~ 

请 注意 ，final 类 的 域 可 以 根据 个 人 的 意愿 选择 为 是 或 不 是 final。 不 论 类 是 否 被 定义 为 final， 
相同 的 规则 都 适用 于 定义 为 final 的 域 。 然 而 ， 由 于 final 类 禁止 继承 ， 所 以 final 类 中 所 有 的 方法 
都 隐 式 指定 为 是 final 的 ， 因 为 无 法 覆盖 它们 。 在 final 类 中 可 以 给 方法 添加 final 修 饰 词 ， 但 这 不 
会 增添 任何 意义 。 

练习 22:(1) 创建 一 个 final 类 并 试 着 继承 它 。 

7.8.4 有 关 final 的 忠告 

在 设计 类 时 ， 将 方法 指明 是 final 的 ， 应 该 说 是 明智 的 。 你 可 能 会 觉得 ， 没 人 会 想 要 覆盖 你 
的 方法 。 有 时 这 是 对 的 。 

但 请 留意 你 所 作 的 假设 。 要 预见 类 是 如 何 被 复 用 的 一 般 是 很 困难 的 ， 特 别 是 对 于 一 个 通用 
类 而 言 更 是 如 此 。 如 果 将 一 个 方法 指定 为 final， 可 能 会 妨碍 其 他 程序 员 在 项 目 中 通过 继承 来 复 
用 你 的 类 ， 而 这 只 是 因为 你 没有 想到 它 会 以 那 种 方式 被 运用 。 

Java 标 准 程序 库 就 是 一 个 很 好 的 例子 。 特 别 是 Java 1.0/1.1 中 Vector 类 被 广泛 地 运用 ， 而 且 从 
效率 考虑 (这 近乎 是 一 个 幻想 ) ， 如 果 所 有 的 方法 均 未 被 指定 为 final 的 话 ， 它 可 能 会 更 加 有 用 。 
很 容易 想像 到 ， 人 们 可 能 会 想 要 继承 并 覆盖 如 此 基础 而 有 用 的 类 ， 但 是 设计 者 却 认为 这 样 做 不 
太 合适 。 这 里 有 两 个 令 人 意外 的 原因 。 第 一 ，Stack 继 承 自 Vector， 就 是 说 Stack 是 个 Vector， 这 
从 逻辑 的 观点 看 是 不 正确 的 。 尽 管 如 此 ，Java 的 设计 者 们 自己 仍旧 继承 了 Vector。 在 以 这 种 方式 
创建 Stack 时 ， 他 们 应 该 意识 到 final 方 法 显得 过 于 严 苛 了 。 

第 二 ，Vector 的 许多 最 重要 的 方法 一 如 addElement0 和 elementAt0 是 同步 的 。 正 如 在 第 21 章 
中 将 要 看 到 的 那样 ， 这 将 导致 很 大 的 执行 开销 ， 可 能 会 抹 煞 final 所 带 来 的 好 处 。 这 种 情况 增强 
了 人 们 关于 程序 员 无 法 正确 猜测 优化 应 当 发 生 于 何 处 的 观点 。 如 此 鉴 脚 的 设计 ， 却 要 置 于 我 们 
每 个 人 都 得 使 用 的 标准 程序 库 中 ， 这 是 很 精 糕 的 〈 幸 运 的 是 ， 现 代 Java 的 容器 库 用 ArrayList 
赫 代 了 Vector。ArrayList 的 行为 要 合理 得 多 。 和 遗憾 的 是 仍然 存在 用 有 旧 容器 库 编写 新 程序 代码 的 
情况 ) 。 

留心 一 下 Hashtable， 这 个 例子 同样 有 趣 ， 它 也 是 一 个 重要 的 Javal.0/1.1 标 准 库 类 ， 而 且 不 
含 任何 final 方 法 。 如 本 书 其 他 地 方 所 提 到 的 ， 某 些 类 明显 是 由 一 些 互 不 相关 的 人 设计 的 (读者 
会 发 现 ， 名 为 Hashtable 的 方法 相对 于 Vector 中 的 方法 要 简洁 得 多 ， 这 又 是 一 个 证 据 )。 对 于 类 库 
的 使 用 者 来 说 ， 这 又 是 一 个 本 不 该 如 此 轻率 的 事物 。 这 种 不 规则 的 情况 只 能 使 用 户 付出 更 多 的 
努力 。 这 是 对 粗糙 的 设计 和 代码 的 又 一 讽刺 〈 请 注意 ， 现 代 Java 的 容器 库 用 HashMap 替 代 了 
Hashtable) 。 


7.9 初始 化 及 类 的 加 载 


在 许多 传统 语言 中 ， 程 序 是 作为 启动 过 程 的 一 部 分 立刻 被 加 载 的 。 然 后 是 初始 化 ， 紧 接着 
程序 开始 运行 。 这 些 语言 的 初始 化 过 程 必须 小 心 控制 ， 以 确保 定义 为 statie 的 东西 ， 其 初始 化 顺 
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序 不 会 造成 麻烦 。 例 如 C++ 中 ， 如 果 某 个 static 期 望 另 一 个 static 在 被 初始 化 之 前 就 能 有 效 地 使 用 
它 ， 那 么 就 会 出 现 问题 。 

Java 就 不 会 出 现 这 个 问题 ， 因 为 它 采 用 了 一 种 不 同 的 加 载 方式 。 加 载 是 众多 变 得 更 加 容易 
的 动作 之 一 ， 因 为 Java 中 的 所 有 事物 都 是 对 象 。 请 记 住 ， 每 个 类 的 编译 代码 都 存在 于 它 自 己 的 
独立 的 文件 中 。 该 文件 只 在 需要 使 用 程序 代码 时 才 会 被 加 载 。 一 般 来 说 ， 可 以 说 :“ 类 的 代码 在 
初次 使 用 时 才 加 载 。” 这 通常 是 指 加 载 发 生 于 创建 类 的 第 一 个 对 象 之 时 ， 但 是 当 访问 static 域 或 
static 方 法 时 ， 也 会 发 生 加 载 。。 

初次 使 用 之 处 也 是 static 初 始 化 发 生 之 处 。 所 有 的 static 对 象 和 static 代 码 段 都 会 在 加 载 时 依 
程序 中 的 顺序 ( 即 ， 定 义 类 时 的 书写 顺序 ) 而 依次 初始 化 。 当 然 ， 定 义 为 static 的 东西 只 会 被 初 
始 化 一 次 。 

7.9.1 继承 与 初始 化 

了 解 包 括 继承 在 内 的 初始 化 全 过 程 ， 以 对 所 发 生 的 一 切 有 个 全 局 性 的 把 握 ， 是 很 有 益 的 。 

请 看 下 例 : 


//: reusing/Beetle.java 
// The full process of initialization. 
import static net.mindview.util.Print.*; 
class Insect { 
private int i = 9; 
protected int j; 
Insect() { 
printis" + at", j=" +5); 
j = 39; 
} 
private static int x1 = 
printInit("static Insect.x1 initialized"); 
static int printinit(String s) { 
print(s); 
return 47; 
š } 


public class Beetle extends Insect { 
private int k = printInit("Beetle.k initialized"); 
public Beetle() { 
print("k = " +k); 
print("j =" + j); 


private static int x2 = 
printInit("static Beetle.x2 initialized"); 
public static void main(String[) args) { 
print("Beetle constructor"); 
Beetle b = new Beetle(); 


} 

/* Output: 

static Insect.x1 initialized 
static Beetle.x2 initialized 
Beetle constructor 


1=9,j=0 
Beetle.k initialized 
k= 47 

j= 39 

Whim 


在 Beetle 上 运行 Java 时 ， 所 发 生 的 第 一 件 事情 就 是 试图 访问 Beetlemain0 (一 个 static 方 法 )， 


O 构造 器 也 是 static 方 法 ， 尽 管 static 关 键 字 并 没有 显 式 地 写 出 来 。 因 此 更 准确 地 讲 ， 关 是 在 其 任何 static 成 员 被 
访问 时 加 载 的 。 





LAR 147 





于 是 加 载 器 开始 启动 并 找 出 Beetie 类 的 编译 代码 (在 名 为 Beetle.class 的 文件 之 中 )。 在 对 它 进行 
加 载 的 过 程 中 ， 编 译 器 注意 到 它 有 一 个 基 类 (这 是 由 关键 字 extends 得 知 的 ) ， 于 是 它 继续 进行 加 
载 。 不 管 你 是 否 打算 产生 一 个 该 基 类 的 对 象 ， 这 都 要 发 生 〈 请 尝试 将 对 象 创建 代码 注释 掉 ， 以 
证 明 这 一 点 ) 。 

如 果 该 基 类 还 有 其 自身 的 基 类 ， 那 么 第 二 个 基 类 就 会 被 加 载 ， 如 此 类 推 。 接 下 来 ， 根 基 类 
中 的 statie 初 始 化 《在 此 例 中 为 Insect) 即 会 被 执行 ， 然 后 是 下 一 个 导出 类 ， 以 此 类 推 。 这 种 方 
式 很 重要 ， 因 为 导出 类 的 static 初 始 化 可 能 会 依赖 于 基 类 成 员 能 否 被 正确 初始 化 。 

至 此 为 止 ， 必 要 的 类 都 已 加 载 完毕 ， 对 象 就 可 以 被 创建 了 。 首 先 ， 对 象 中 所 有 的 基本 类 型 
都 会 被 设 为 默认 值 , 对 象 引用 被 设 为 null 一 这 是 通过 将 对 象 内 存 设 为 二 进 制 零 值 而 一 举 生成 的 。 
然后 ， 基 类 的 构造 器 会 被 调用 。 在 本 例 中 ， 它 是 被 自动 调用 的 。 但 也 可 以 用 super 来 指定 对 基 类 
构造 器 的 调用 (正如 在 Beetle0 构 造 器 中 的 第 一 步 操作 )。 基 类 构造 器 和 导出 类 的 构造 器 一 样 ， 
以 相同 的 顺序 来 经 历 相同 的 过 程 。 在 基 类 构造 器 完成 之 后 ， 实 例 变量 按 其 次 序 被 初始 化 。 最 后 ， 
构造 器 的 其 余部 分 被 执行 。 

练习 23:， (2) 请 证 明 加 载 类 的 动作 仅 发 生 一 次 。 证 明 该 类 的 第 一 个 实体 的 创建 或 者 对 static 
成 员 的 访问 都 有 可 能 引起 加 载 。 

练习 24， (2) 在 Beetlejava 中 ， 从 Beetle 类 继承 产生 一 个 具体 类 型 的 “甲壳 虫 "。 其 形式 与 现 
有 类 相同 ， 跟 踪 并 解释 其 输出 结果 。 


7.10 总 结 


继承 和 组 合 都 能 从 现 有 类 型 生成 新 类 型 。 组 合 一 般 是 将 现 有 类 型 作为 新 类 型 底层 实现 的 一 
部 分 来 加 以 复 用 ， 而 继承 复 用 的 是 接口 。 

在 使 用 继承 时 ， 由 于 导出 类 具有 基 类 接口 ， 因 此 它 可 以 向 上 转型 至 基 类 ， 这 对 多 态 来 讲 至 
关 重 要 ， 就 像 我 们 将 在 下 一 章 中 将 要 看 到 的 那样 。 

尽管 面向 对 象 编程 对 继承 极力 强调 ， 但 在 开始 一 个 设计 时 ， 一 般 应 优先 选择 使 用 组 合 (或 
者 可 能 是 代理 )， 只 在 确实 必要 时 才 使 用 继承 。 因 为 组 合 更 具 灵 活性 。 此 外 ， 通 过 对 成 员 类 型 使 
用 继承 技术 的 添加 技巧 ， 可 以 在 运行 时 改变 那些 成 员 对 象 的 类 型 和 行为 。 因 此 ， 可 以 在 运行 时 
改变 组 合 而 成 的 对 象 的 行为 。 

在 设计 一 个 系统 时 ， 目 标 应 该 是 找到 或 创建 某 些 类 ， 其 中 每 个 类 都 有 具体 的 用 途 ， 而 且 既 
不 会 太 大 (包含 太 多 的 功能 而 难以 复 用 )， 也 不 会 太 小 (不 添加 其 他 功能 就 无 法 使 用 )。 如 果 你 
的 设计 变 得 过 于 复杂 ， 通 过 将 现 有 类 拆 分 为 更 小 的 部 分 而 添加 更 多 的 对 象 ， 通 常会 有 所 帮助 。 

当 你 开始 设计 一 个 系统 时 ， 应 该 认识 到 程序 开发 是 一 种 增 量 过 程 ， 犹 如 和 类 的 学 习 一 样 ， 
这 一 点 很 重要 。 程 序 开发 依赖 于 实验 ， 你 可 以 尽 己 所 能 去 分 析 ， 但 当 你 开始 执行 一 个 项 目 时 ， 
你 仍然 无 法 知道 所 有 的 答案 。 如 果 将 项 目 视 作 是 一 种 有 机 的 、 进 化 着 的 生命 体 而 去 培养 ， 而 不 
是 打算 像 盖 摩 天 大 楼 一 样 快速 见效 ， 就 会 获得 更 多 的 成 功 和 更 迅速 的 回馈 。 继 承 与 组 合 正 是 在 
面向 对 象 程序 设计 中 使 得 你 可 以 执行 这 种 实验 的 最 基本 的 两 个 工具 。 

所 选 习 题 的 答案 都 可 以 在 名 为 The Thinking in Java Annotated Solution Guide 的 电子 文档 中 找 
到 ,读者 可 以 从 www.MindView.net 购 买 此 文档 。 
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第 8 章 多 T 


“我 曾经 被 问 到 “求教 ，Babbage 先 生 ， 如 果 你 向 机 器 中 输入 错误 的 数字 ， 可 以 得 到 正确 的 
答案 吗 ? ”我 无 法 恰当 地 理解 产生 这 种 问题 的 概念 上 的 混淆 ” 

Charles Babbage(1791-1871) 

在 面向 对 象 的 程序 设计 语言 中 ， 多 态 是 继 数据 抽象 和 继承 之 后 的 第 三 种 基本 特征 。 

多 态 通 过 分 离 做 什么 和 怎么 做 ， 从 另 一 角度 将 接口 和 实现 分 离开 来 。 多 态 不 但 能 够 改善 代 
码 的 组 织 结构 和 可 读 性 ， 还 能 够 创建 可 扩展 的 程序 一 一 即 无 论 在 项 目 最 初创 建 时 还 是 在 需要 添 
加 新 功能 时 都 可 以 “生长 ”的 程序 。 

“封装 ”通过 合并 特征 和 行为 来 创建 新 的 数据 类 型 。“ 实 现 隐 藏 ” 则 通过 将 细节 “私有 化 ” 
把 接口 和 实现 分 离开 来 。 这 种 类 型 的 组 织 机 制 对 那些 拥有 过 程 化 程序 设计 背景 的 人 来 说 ， 更 容 
易 理解 。 而 多 态 的 作用 则 是 消除 类 型 之 间 的 耦合 关系 。 在 前 一 章 中 我 们 已 经 知道 ， 继 承 允 许 将 
对 象 视 为 它 自 己 本 身 的 类 型 或 其 基 类 型 来 加 以 处 理 。 这 种 能 力 极为 重要 ， 因 为 它 允 许 将 多 种 类 
型 (从 同一 基 类 导出 的 ) 视 为 同一 类 型 来 处 理 ， 而 同一 份 代码 也 就 可 以 毫 无 差别 地 运行 在 这 些 
不 同类 型 之 上 了 。 多 态 方 法 调用 允许 一 种 类 型 表现 出 与 其 他 相似 类 型 之 间 的 区 别 ， 只 要 它们 都 
是 从 同一 基 类 导出 而 来 的 。 这 种 区 别 是 根据 方法 行为 的 不 同 而 表示 出 来 的 ， 虽 然 这 些 方法 都 可 
以 通过 同一 个 基 类 来 调用 。 

在 本 章 中 ， 通 过 一 些 基 本 、 简 单 的 例子 这些 例子 中 所 有 与 多 态 无 关 的 代码 都 被 删 掉 ， 只 
剩 下 与 多 态 有 关 的 部 分 )， 深入浅出 地 介绍 多 态 〈 也 称 作 动态 绑 定 、 后 期 绑 定 或 运行 时 绑 定 ) 。 


8.1 再 论 向 上 转型 


在 第 7 章 中 我 们 已 经 知道 ， 对 象 既 可 以 作为 它 自己 本 身 的 类 型 使 用 ， 也 可 以 作为 它 的 基 类 型 
使 用 。 而 这 种 把 对 某 个 对 象 的 引用 视 为 对 其 基 类 型 的 引用 的 做 法 被 称 作 向 上 转型 一 因为 在 继 
承 树 的 画 法 中 ， 基 类 是 放置 在 上 方 的 。 

但 是 ， 这 样 做 也 有 一 个 问题 ， 具 体 看 下 面 这 个 有 关 乐 器 的 例子 。 

首先 ， 既 然 几 个 例子 都 要 演奏 乐 符 (Note) ， 我 们 就 应 该 在 包 中 单独 创建 一 个 Note 类 。 


/1: polymorphism/music/Note. java 
1/ Notes to play on musical instruments. 
package polymorphism.music; 


public enum Note { 
MIDDLE_C, C_SHARP, B_FLAT; // Etc. 
} AAA 


enum 在 第 5 章 中 介绍 过 。 
在 这 里 ，Wind 是 一 种 Instrument， 因 此 可 以 从 Instrument 类 继承 。 


//: polymorphism/music/Instrument. java 
‘package polymorphism.music; 
import static net.mindview.util.Print.*; 


class Instrument { 
public void play(Note n) { 
print ("Instrument. play()"); 
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//: polymorphism/music/Wind. java 
package polymorphism.music; 


// Wind objects are instruments 
// because they have the same interface: 
public class Wind extends Instrument { 
// Redefine interface method: 
public void play(Note n) { 


System.out.printin("Wind.play() " + n); 


} 
} 4:~ 


1/: polymorphism/music/Music. java 
// Inheritance & upcasting. 
package polymorphism.music; 


public class Music { 
public static void tune(Instrument 1) { 
OF ase 


i play (Note. MIDDLE_C); 
public static void main(String(] args) { 
Wind flute = new Wind(); 
tune(flute); // Upcasting 
i 
} /* Output: 
Wind.play() MIDDLE_C 
Whim 


Music.tune() 方 法 接受 一 个 Instrument 引 用 ， 同 时 也 接受 任何 导出 自 Instrument 的 类 。 在 


main0 方 法 中 ， 当 一 个 Wind 引 用 传递 到 tune0 方 法 时 ， 就 会 出 现 这 种 情况 ， 而 不 需要 任何 类 型 
转换 。 这 样 做 是 允许 的 一 因为 Wind 从 Instrument 继 承 而 来 ， 所 以 Instrument 的 接口 必定 存在 
于 Wind 中 。 从 Wind 向 上 转型 到 Instrument 可 能 会 “缩小 ”接口 ， 但 不 会 比 Instrument 的 全 部 接 
DER, 

8.1.1 忘记 对 象 类 型 


Musicjava 看 起 来 似乎 有 些 奇怪 。 为 什么 所 有 人 都 故意 忘记 对 象 的 类 型 呢 ? 在 进行 向 上 转型 


时 ， 就 会 产生 这 种 情况 ， 并 且 如 果 让 tune0 方 法 直接 接受 一 个 Wind 引 用 作为 自己 的 参数 ， 似 乎 
会 更 为 直观 。 但 这 样 引发 的 一 个 重要 问题 是 : 如 果 那 样 做 ， 就 需要 为 系统 内 Instrument 的 每 种 
类 型 都 编写 一 个 新 的 tune0 方 法 。 假 设 按照 这 种 推理 ， 现 在 再 加 入 Stringed ( 弦 乐 ) 和 Brass ( 管 
乐 ) 这 两 种 Instrument (乐器 ) : 


/1: polymorphism/music/Music2.java . 
// Overloading instead of upcasting. 
package polymorphism. music: 
import static net.mindview.util.Print.*; 
class Stringed extends Instrument { 

public void play(Note n) { 

print (*Stringed.play() * + n); 

} 

} 


class Brass extends Instrument { 
public void play(Note n) { 
print("Brass.play() " + n); 
} 
} 


public class Music2 { 
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public static void tune(Wind i) { 
i .play (Note .MIDDLE_C) ; 

} 

public static void tune(Stringed i) { 
i .play (Note. MIDDLE_C) ; 


public static void tune(Brass i) { 
i.play(Note.MIODLE_C); 

} 

public static void main(String{] args) { 
Wind flute = new Wind(): 
Stringed violin = new Stringed(): 
Brass frenchHorn = new Brass(); 
tune(flute); // No upcasting 
tune (violin); 
tune(frenchHorn) ; 


} 
} /* Output: 
Wind. play() MIDDLE_C 
Stringed. play() MIDDLE_C 
Brass.play() MIDDLE_C 
Wim 


这 样 做 行 得 通 ， 但 有 一 个 主要 缺点 ， 必 须 为 添加 的 每 一 个 新 Instrument 类 编写 特定 类 型 的 
方法 。 这 意味 着 在 开始 时 就 需要 更 多 的 编程 ， 这 也 意味 着 如 果 以 后 想 添加 类 似 tune0 的 新 方法 ， 
或 者 添加 自 Instrument 导 出 的 新 类 ， 仍 需要 做 大 量 的 工作 。 此 外 ， 如 果 我 们 忘记 重 载 某 个 方法 ， 
编译 器 不 会 返回 任何 错误 信息 ， 这 样 关 于 类 型 的 整个 处 理 过 程 就 变 得 难以 操纵 。 

如 果 我 们 只 写 这 样 一 个 简单 方法 ， 它 仅 接收 基 类 作为 参数 ， 而 不 是 那些 特殊 的 导出 类 。 这 
样 做 情况 会 变 得 更 好 吗 ? 也 就 是 说 ， 如 果 我 们 不 管 导出 类 的 存在 ， 编 写 的 代码 只 是 与 基 类 打 交 
道 ， 会 不 会 更 好 呢 ? 

这 正 是 多 态 所 允许 的 。 然 而 ， 大 多 数 程序 员 具 有 面向 过 程 程序 设计 的 背景 ， 对 多 态 的 运作 
方式 可 能 会 有 一 点 迷惑 。 

练习 1，(2) 创建 一 个 Cycle 类 ， 它 具有 子 类 Unicycle、Bicycle 和 Tricycle。 演 示 每 一 个 类 型 的 
实例 都 可 以 经 由 ride0 方 法 向 上 转型 为 Cycle。 


8.2 转机 


运行 这 个 程序 后 ， 我 们 便 会 发 现 Music.java 的 难点 所 在 。Wind.play0 方 法 将 产生 输出 结果 。 
这 无 疑 是 我 们 所 期 望 的 输出 结果 ， 但 它 看 起 来 似乎 又 没有 什么 意义 。 请 观察 一 下 tune0 方 法 : 


public static void tune(Instrument i) { 
DE ss 


i.play (Note. MIDDLE_C) ; 
} 


它 接受 一 个 Instrument 引 用 。 那 么 在 这 种 情况 下 ， 编 译 器 怎样 才能 知道 这 个 Instrument 引 用 指向 
的 是 Wind 对 象 ， 而 不 是 Brass 对 象 或 Stringed 对 象 呢 ? 实际 上 ， 编 译 器 无 法 得 知 。 为 了 深入 理 解 
这 个 问题 ， 有 必要 研究 一 下 绑 定 这 个 话题 。 
8.2.1 方法 调用 绑 定 

将 一 个 方法 调用 同一 个 方法 主体 关联 起 来 被 称 作 绑 定 。 若 在 程序 执行 前 进行 绑 定 〈( 如 果 有 
的 话 ， 由 编译 器 和 连接 程序 实现 ) ， 叫 做 前 期 绑 定 。 读 者 可 能 以 前 从 来 没有 听 说 过 这 个 术语 ， 因 
为 它 是 面向 过 程 的 语言 中 不 需要 选择 就 默认 的 绑 定 方式 。 例 如 ，C 只 有 一 种 方法 调用 ， 那 就 是 前 
期 绑 定 。 
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上 述 程序 之 所 以 令 人 迷惑 ， 主 要 是 因为 前 期 绑 定 。 因 为 ， 当 编译 器 只 有 一 个 Instrument 引 
用 时 ， 它 无 法 知道 究竟 调用 哪个 方法 才 对 。 

解决 的 办 法 就 是 后 期 绑 定 ， 它 的 含义 就 是 在 运行 时 根据 对 象 的 类 型 进行 绑 定 。 后 期 绑 定 也 
则 做 动态 绑 定 或 运行 时 绑 定 。 如 果 一 种 语言 想 实现 后 期 绑 定 ， 就 必须 具有 某 种 机 制 ， 以 便 在 运 
行 时 能 判断 对 象 的 类 型 ， 从 而 调用 恰当 的 方法 。 也 就 是 说 ， 编 译 器 一 直 不 知道 对 象 的 类 型 ， 但 
是 方法 调用 机 制 能 找到 正确 的 方法 体 ， 并 加 以 调用 。 后 期 绑 定 机 制 随 编程 语言 的 不 同 而 有 所 不 
同 ， 但 是 只 要 想 一 下 就 会 得 知 ， 不 管 怎样 都 必须 在 对 象 中 安置 某 种 “类 型 信息 "。 

Java 中 除了 static 方 法 和 final 方 法 (private 方 法 属于 final 方 法 ) 之 外 ， 其 他 所 有 的 方法 都 是 
后 期 绑 定 。 这 意味 着 通常 情况 下 ， 我 们 不 必 判 定 是 否 应 该 进行 后 期 绑 定 -一 它 会 自动 发 生 。 

为 什么 要 将 某 个 方法 声明 为 final 呢 ?正如 前 一 章 提 到 的 那样 ， 它 可 以 防止 其 他 人 覆盖 该 方 
法 。 但 更 重要 的 一 点 或 许 是 : 这 样 做 可 以 有 效 地 “关闭 ”动态 绑 定 ， 或 者 说 ， 告 诉 编译 器 不 需 
要 对 其 进行 动态 绑 定 。 这 样 ， 编 译 器 就 可 以 为 final 方 法 调用 生成 更 有 效 的 代码 。 然而 ， 大 多 数 
情况 下 ， 这 样 做 对 程序 的 整体 性 能 不 会 有 什么 改观 。 所 以 ， 最 好 根据 设计 来 决定 是 否 使 用 final， 
而 不 是 出 于 试图 提高 性 能 的 目的 来 使 用 final。 

8.2.2 产生 正确 的 行为 

一 且 知 道 Java 中 所 有 方法 都 是 通过 动态 绑 定 实现 多 态 这 个 事实 之 后 ， 我 们 就 可 以 编写 只 与 
基 类 打交道 的 程序 代码 了 ， 并 且 这 些 代码 对 所 有 的 导出 类 都 可 以 正确 运行 。 或 者 换 一 种 说 法 ， 
发 送 消息 给 某 个 对 象 ， 让 该 对 象 去 断定 应 该 做 什么 事 。 

面向 对 象 程序 设计 中 ， 有 一 个 经 典 的 例子 就 是 “几何 形状 ”(shape) 。 因 为 它 很 直观 ， 所 以 
经 常用 到 ， 但 不 幸 的 是 ， 它 可 能 使 初学 者 认为 面向 对 象 程序 设计 仅 适 用 于 图 形 化 程序 设计 ， 实 
际 当然 不 是 这 样 。 

在 “几何 形状 ”这 个 例子 中 ， 有 一 个 基 类 Shape， 以 及 多 个 导出 类 一 一 如 Circle、Square、 
Triangle 等 。 这 个 例子 之 所 以 好 用 ， 是 因为 我 们 可 以 说 “ 贺 是 一 种 几何 形状 "， 这 种 说 法 也 很 容 
易 被 理解 。 下 面 的 继承 图 展示 它们 之 间 的 关系 : 





A Shape 
向 上 转型 继承 图 draw() 
erase() 









































i Circle Square Triangle 
Circle draw() draw() draw() 
Reference} | erase() erase() erase() 














向 上 转型 可 以 像 下 面 这 条 语句 这 么 简单 : 

Shape s = new Circle(); 

这 里 ， 创 建 了 一 个 Circle 对 象 ， 并 把 得 到 的 引用 立即 赋值 给 Shape， 这 样 做 看 似 错误 (将 一 
种 类 型 赋值 给 另 一 种 类 型 ) ， 但 实际 上 是 没 问题 的 ， 因 为 通过 继承 ，Circle 就 是 一 种 Shape。 因 
此 ， 编 译 器 认可 这 条 语句 ， 也 就 不 会 产生 错误 信息 。 

假设 你 调用 一 个 基 类 方法 〈 它 已 在 导出 类 中 被 覆盖 ) : 


s.draw(); 
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你 可 能 再 次 认为 调用 的 是 Shape 的 draw0， 因 为 这 毕竟 是 一 个 Shape 引 用 ， 那 么 编译 器 是 怎样 知 
道 去 做 其 他 的 事情 呢 ? 由 于 后 期 绑 定 〈 多 态 ) ， 还 是 正确 调用 了 Circle.draw0 方 法 。 
下 面 的 例子 稍微 有 所 不 同 : 


/1/: polymorphism/shape/Shape. java 
package polymorphism. shape; 





public class Shape { 
public void draw() {} 
public void erase() {} 
[283] y Mi~ 
//: polymorphism/shape/Circle. java 
package polymorphism. shape: 
import static net.mindview.util.Print.*; 














public class Circle extends Shape { 
public void draw() { print("Circle.draw()"); } 
public void erase() { print("Circle.erase()"); } 
} Mi~ 


11: polymorphism/shape/Square. java 
package polymorphism. shape; 
import static net.mindview.util.Print.*; 


public class Square extends Shape { 
public void draw() { print("Square.draw()"); 
public void erase() { print("Square.erase()") 
} Mm 
//: polymorphism/shape/Triangle. java 


package polymorphism. shape; 
import static net.mindview.util.Print.*; 





public class Triangle extends Shape { 
public void draw() { print("Triangle.draw()") 
public void erase() { print("Triangle.erase()* 
} Mi~ 


11: polymorphism/shape/RandomShapeGenerator . java 
// A "factory" that randomly creates shapes. 
package polymorphism. shape: 

import java.util.*; 





public class RandomShapeGenerator { 
private Random rand = new Random(47) ; 
public Shape next() { 
switch(rand.nextInt(3)) { 
default: 
case @: return new Circle(): 
case 1: return new Square(); 
case 2: return new Triangle(): 
} 
} 
} Mi~ 


/1: polymorphism/Shapes. java 
/1 Polymorphism in Java. 


import polymorphism. shape.* 








public class Shapes { 

private static RandomShapeGenerator gen = 
new RandomShapeGenerator () ; 

public static void main(String{] args) { 
Shape[] s = new Shape[9]; 
7/ Fill up the array with shapes: 
for(int i = @; i < s.length; i++) 

s[i] = gen.next(); 
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// Make polymorphic method calls: 
for(Shape shp : s) 
shp.draw() ; 


} 
} /* Output: 
Triangle.draw() 
Triangle. draw() 
Square.draw() 
Triangle.draw() 
Square.draw() 
Triangle.draw() 
Square.draw() 
Triangle.draw() 
Circle.draw() 
“i~ 


Shape 基 类 为 自 它 那里 继承 而 来 的 所 有 导出 类 建立 了 一 个 公用 接口 一 也 就 是 说 ， 所 有 形 
状 都 可 以 描绘 和 控 除 。 导 出 类 通过 覆盖 这 些 定 义 ， 来 为 每 种 特殊 类 型 的 几何 形状 提供 单独 的 
行为 。 

RandomShapeGenerator 是 一 种 “工厂 ” (factory) ， 在 我 们 每 次 调用 next0 方 法 时 ， 它 可 以 
为 随机 选择 的 Shape 对 象 产生 一 个 引用 。 请 注意 向 上 转型 是 在 return 语 句 里 发 生 的 。 每 个 return 
语句 取得 一 个 指向 某 个 Circle、Square 或 者 Triangle 的 引用 ， 并 将 其 以 Shape 类 型 从 next0 方 法 中 
发 送出 去 。 所 以 无 论 我 们 在 什么 时 候 调 用 nextO 方 法 时 ， 是 绝对 不 可 能 知道 具体 类 型 到 底 是 什么 
的 ， 因 为 我 们 总 是 只 能 获得 一 个 通用 的 Shape 引 用 。 

main(0 包 含 了 一 个 Shape 引 用 组 成 的 数组 ， 通 过 调用 RandomShapeGeneratornext0 来 填 人 
数据 。 此 时 ， 我 们 只 知道 自己 拥有 一 些 Shape， 除 此 之 外 不 会 知道 更 具体 的 情况 (编译 器 也 不 知 
道 )。 然 而 ， 当 我 们 遍历 这 个 数组 ， 并 为 每 个 数组 元 素 调用 draw0 方 法 时 ， 与 类 型 有 关 的 特定 行 
为 会 神奇 般 地 正确 发 生 ， 我 们 可 以 从 运行 该 程序 时 所 产生 的 输出 结果 中 发 现 这 一 点 。 

随机 选择 几何 形状 是 为 了 让 大 家 理解 : 在 编译 时 ， 编 译 器 不 需要 获得 任何 特殊 信息 就 能 进 
行 正确 的 调用 。 对 draw(0 方 法 的 所 有 调用 都 是 通过 动态 绑 定 进行 的 。 

练习 2: (1) 在 几何 图 形 的 示例 中 添加 @Override 注 解 。 

练习 3 (1) 在 基 类 Shapes.java 中 添加 一 个 新 方法 ， 用 于 打印 一 条 消息 ， 但 导出 类 中 不 要 各 
盖 这 个 方法 。 请 解释 发 生 了 什么 。 现 在 ， 在 其 中 一 个 导出 类 中 覆盖 该 方法 ， 而 在 其 他 的 导出 类 
中 不 予 覆 盖 ， 观 察 又 有 什么 发 生 。 最 后 ， 在 所 有 的 导出 类 中 摄 盖 这 个 方法 。 

练习 4，(2) 向 Shapesjava 中 添加 一 个 新 的 Shape 类 型 ， 并 在 main0 方 法 中 验证 ， 多 态 对 新 类 
型 的 作用 是 否 与 在 旧 类 型 中 的 一 样 。 

BAIS: (1) 以 练习 1 为 基础 ， 在 Cycle 中 添加 wheels0 方 法 ， i aida 修改 ride0 
方法 ， 让 它 调 用 wheels0 方 法 ， 并 验证 多 态 起 作用 了 。 

8.2.3 可 扩展 性 

现在 ， 让 我 们 返回 到 “乐器 ”(Instrument) 示例 。 由 于 有 多 态 机 制 ， 我 们 可 根据 自己 的 需 
求 对 系统 添加 任意 多 的 新 类 型 ， 而 不 需 更 改 tune0 方 法 。 在 一 个 设计 良好 的 OOP 程 序 中 ， 大 多 数 
或 者 所 有 方法 都 会 遵循 tune0 的 模型 ， 而 且 只 与 基 类 接口 通信 。 这 样 的 程序 是 可 扩展 的 ， 因 为 可 
以 从 通用 的 基 类 继承 出 新 的 数据 类 型 ， 从 而 新 添 一 些 功能 。 那 些 操纵 基 类 接口 的 方法 不 需要 任 
何 改动 就 可 以 应 用 于 新 类 。 

考虑 一 下 : 对 于 “乐器 ”的 例子 ， 如 果 我 们 向 基 类 中 添加 更 多 的 方法 ， 并 加 入 一 些 新 类 ， 
将 会 出 现 什么 情况 呢 ? 请 看 下 图 : 
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Instrument 





void play() 
String what() 
void adjust() 


一 一 十 一 一 





























wind Percussion | Stringed | 
void play() void play() void play() 
String what() | | string what() | | string what() 
void adjust() | | void adjust() | |void adjust) 
po 
Pree) ee eee 全 下 
Woodwind Brass | 
| void play) void play() | 
String what() | | void adjust) | 











事实 上 ， 不 需要 改动 tune( 方 法 ， 所 有 的 新 类 都 能 与 原 有 类 一 起 正确 运行 。 即 使 me0 方 法 
是 单独 存放 在 某 个 文件 中 ， 并 且 在 Instrument 接 口中 添加 了 其 他 的 新 方法 ，tune0 也 不 需 再 编译 
就 能 正确 运行 。 下 面 是 上 图 的 具体 实现 : 


//: polymorphism/music3/Music3. java 
17 An extensible program. 

package polymorphism.music3; 

import polymorphism.music.Note; 

import static net.mindview.util.Print.*; 





class Instrument { 
void play(Note n) { print("Instrument.play() " + n); } 
String what() { return "Instrument"; } 
void adjust() { print("Adjusting Instrument"); } 

} 


class Wind extends Instrument ( 
void play(Note n) { print("Wind.play() " +n); } 
String what() { return "Wind": } 
void adjust() { print("Adjusting Wind"); } 

} 





class Percussion extends Instrument { 
void play(Note n) { print("Percussion.play() " + n); } 
String what() { return "Percussion"; } 
void adjust() { print("Adjusting Percussion"); 
2 


class Stringed extends Instrument { 
void play(Note n) { print("Stringed.play() "+ n); } 
String what() { return "Stringed"; } 
void adjust() { print("Adjusting Stringed"); } 

} 





class Brass extends Wind { 
void play(Note n) { print("Brass.play() " + n); } 
void adjust() { print("Adjusting Brass"); } 

} 


class Woodwind extends Wind { 
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void play(Note n) { print("Woodwind.play() "+ n); } 
String what() { return "Woodwind"; } 
} 


public class Music3 { 
// Doesn't care about type, so new types 
// added to the system still work right: 
public static void tune(Instrument i) { 
PE s3 
i.play (Note .MIDDLE_C) ; 
} 


public static void tuneAll(Instrument[] e) { 
for(Instrument i : e) 
tune(i); 
} 
public static void main(String[] args) { 
// Upcasting during addition to the array: 
Instrument {] orchestra = { 
new Wind(), 
new Percussion(), 
new Stringed(), 
new Brass(), 
new Woodwind() 


}; 
tuneAll (orchestra); 


} 
} /* Output: 
Wind. play() MIDDLE_C 
Percussion.play() MIDDLE_C 
Stringed. play() MIDDLE_C 
Brass.play() MIDDLE_C 
Woodwind. play() MIDDLE_C 
Whim 


新 添加 的 方法 what0 返 回 一 个 带 有 类 描述 的 String 引 用 ， 另 一 个 新 添加 的 方法 adjust() 则 提 
供 每 种 乐器 的 调 音 方法 。 

在 main0 中 ， 当 我 们 将 某 种 引用 置 入 orchestra 数 组 中 ， 就 会 自动 向 上 转型 到 Instrument。 

可 以 看 到 ，tune() 方 法 完全 可 以 忽略 它 周 围 代码 所 发 生 的 全 部 变化 ， 依 旧 正 常 运行 。 这 正 
是 我 们 期 望 多 态 所 具有 的 特性 。 我 们 所 做 的 代码 修改 ， 不 会 对 程序 中 其 他 不 应 受到 影响 的 部 
分 产生 破坏 。 换 句 话说 ， 多 态 是 一 项 让 程序 员 “ 将 改变 的 事物 与 未 变 的 事物 分 离开 来 ”的 重 
要 技术 。 

练习 6: (1) 修改 Musie3.java， 使 what() 方 法 成 为 根 Object 的 toString() 方 法 。 试 用 
System.out.printin0 方 法 打印 Instrument 对 象 (不 用 向 上 转型 ) 。 

练习 7: (2) 向 Music3.java 添 加 一 个 新 的 类 型 Instrument， 并 验证 多 态 性 是 否 作用 于 所 添加 
的 新 类 型 。 

练习 8: (2) 修改 Music3.java， 使 其 可 以 像 Shapes.java 中 的 方式 那样 随机 创建 Instrument 
对 象 。 

练习 9，(3) 创建 Rodent ( 哮 齿 动物 ) : Mouse (老鼠 ) Gerbil (BER), Hamster (大 烦 鼠 )， 
等 等 这 样 一 个 的 继承 层次 结构 。 在 基 类 中 ， 提 供 对 所 有 的 Rodent 都 通用 的 方法 ， 在 导出 类 中 ， 
根据 特定 的 Rodent 类 型 覆盖 这 些 方法 ， 以 便 它们 执行 不 同 的 行为 。 创 建 一 个 Robent 数 组 ， 填 充 
不 同 的 Rodent 类 型 ， 然 后 调用 基 类 方法 ， 观 察 发 生 什么 情况 。 

练习 10: (3) 创建 一 个 包含 两 个 方法 的 基 类 。 在 第 一 个 方法 中 可 以 调用 第 二 个 方法 。 然 后 产 
生 一 个 继承 自 该 基 类 的 导出 类 ， 且 覆盖 基 类 中 的 第 二 个 方法 。 为 该 导出 类 创建 一 个 对 象 ， 将 它 
向 上 转型 到 基 类 型 并 调用 第 一 个 方法 ， 解 释 发 生 的 情况 。 
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8.2.4 缺陷 :“ 和 覆盖 ”私有 方法 
我 们 试图 像 下 面 这 样 做 也 是 无 可 厚 非 的 : 


11: polymorphism/PrivateOverride.java 
// Trying to override a private method. 
package polymorphism; 

import static net.mindview.util.Print.*; 


public class PrivateOverride { 
private void f() { print("private fO"); } 
public static void main(String{] args) { 
PrivateOverride po = new Derived(): 
po. FQ); 
} 
$ 


class Derived extends PrivateOverride { 
public void f() { print("public fO"); } 

} /* Output: 

private fO 

Whim 


我 们 所 期 望 的 输出 是 public f0， 但 是 由 于 private 方 法 被 自动 认为 是 final 方 法 ， 而 且 对 导出 
类 是 屏蔽 的 。 因 此 ， 在 这 种 情况 下 ，Derived 类 中 的 f0 方 法 就 是 一 个 全 新 的 方法 ， 既 然 基 类 中 的 


10 方 法 在 子 类 Derived 中 不 可 见 ， 因 此 甚至 也 不 能 被 重 载 。 


结论 就 是 ， 只 有 非 private 方 法 才 可 以 被 覆盖 ， 但 是 还 需要 密切 注意 覆盖 private 方 法 的 现象 ， 
这 时 虽然 编译 器 不 会 报错 ， 但 是 也 不 会 按照 我 们 所 期 望 的 来 执行 。 确 切 地 说 ， 在 导出 类 中 ， 对 


于 基 类 中 的 private 方 法 ， 最 好 采用 不 同 的 名 字 。 
8.2.5 缺陷 ， 域 与 静态 方法 


一 旦 你 了 解 了 多 态 机 制 ， 可 能 就 会 开始 认为 所 有 事物 都 可 以 多 态 地 发 生 。 然 而 ， 只 有 普通 


的 方法 调用 可 以 是 多 态 的 。 例 如 ， 如 果 你 直接 访问 某 个 域 ， 这 个 访问 就 将 在 编译 期 进行 解析 ， 


就 像 下 面 的 示例 所 演示 的 。 : 


//: polymorphism/FieldAccess. java 


// Direct field access is determined at compile time. 


class Super { 

public int field = 0; 

public int getField() { return field: } 
} 


class Sub extends Super { 

public int field = 1; 

public int getField() { return field; } 

public int getSuperField() { return super.field: } 
} 


public class FieldAccess { 
public static void main(String(] args) { 
Super sup = new Sub(); // Upcast 
System.out.printin(“sup.field = " + sup.field: + 
*, sup.getField() = ”+ sup.getField()); 
Sub sub = new Sub(); 
System.out.printIn("sub. field = ”+ 
sub.field + ", sub.getField() = ”+ 
sub.getField() + 
", sub.getSuperField() = ”+ 
sub. getSuperField()); 


© 感谢 Randy Nichols 提 出 了 这 个 问题 。 
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} 
} /* Output: 
sup.field = 0, sup.getField() = 1 
sub. field = 1, sub.getField() = 1, sub.getSuperField() = 9 
Whim 


当 Sub 对 象 转型 为 Super 引 用 时 ， 任 何 域 访问 操作 都 将 由 编译 器 解析 ， 因 此 不 是 多 态 的。 在 
本 例 中 ， 为 Superfield 和 Sub.field 分 配 了 不 同 的 存储 空间 。 这 样 ，Sub 实 际 上 包含 两 个 称 为 field 
的 域 : 它 自己 的 和 它 从 Super 处 得 到 的 。 然 而 ， 在 引用 Sub 中 的 field 时 所 产生 的 默认 域 并 非 Super 
版 本 的 field 域 。 因 此 ， 为 了 得 到 Superfield， 必 须 显 式 地 指明 superfield 。 

尽管 这 看 起 来 好 像 会 成 为 一 个 容易 令 人 混 清 的 问题 ， 但 是 在 实践 中 ， 它 实际 上 从 来 不 会 发 
生 。 首 先 ， 你 通常 会 将 所 有 的 域 都 设置 成 private， 因 此 不 能 直接 访问 它们 ， 其 副作用 是 只 能 调 
用 方法 来 访问 。 另 外 ， 你 可 能 不 会 对 基 类 中 的 域 和 导出 类 中 的 域 赋予 相同 的 名 字 ， 因 为 这 种 做 
法 容易 令 人 混 消 。 

如 果 某 个 方法 是 静态 的 ， 它 的 行为 就 不 具有 多 态 性 : 

//: polymorphism/StaticPolymorphism, java 

/1 Static methods are not polymorphic. 


class StaticSuper { 
public static String staticGet() { 
return "Base staticGet()"; 


} 
public String dynamicGet() { 
return "Base dynamicGet()"; 
} 
} 


class StaticSub extends StaticSuper { 
public static String staticGet() { 
return "Derived staticGet()"; 


} 
public String dynamicGet() { 
return "Derived dynamicGet()"; 
} 
} 


public class StaticPolymorphism { 
public static void main(String{] args) { 
StaticSuper sup = new StaticSub(); // Upcast 
System.out.printIn(sup.staticGet()); 
System. out. printin(sup.dynamicGet()) ; 


} /* Output: 

Base staticGet() 
Derived dynamicGet() 
Whim 


静态 方法 是 与 类 ， 而 并 非 与 单个 的 对 象 相 关联 的 。 
8.3 构造 器 和 多 态 

通常 ， 构 造 器 不 同 于 其 他 种 类 的 方法 。 涉 及 到 多 态 时 仍 是 如 此 。 尽 管 构造 器 并 不 具有 多 态 
性 〈 它 们 实际 上 是 static 方 法 ， 只 不 过 该 static 声 明 是 隐 式 的 ) ， 但 还 是 非常 有 必要 理解 构造 器 怎 
样 通过 多 态 在 复杂 的 层次 结构 中 运作 ， 这 一 理解 将 有 助 于 大 家 避免 一 些 令 人 不 快 的 困扰 。 
8.3.1 构造 器 的 调用 顺序 


构造 器 的 调用 顺序 已 在 第 5 章 进行 了 简要 说 明 ， 并 在 第 7 章 再 次 提 到 ， 但 那些 都 是 在 多 态 引 
入 之 前 介绍 的 。 
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基 类 的 构造 器 总 是 在 导出 类 的 构造 过 程 中 被 调用 ， 而 且 按照 继承 层次 逐渐 向 上 链接 ， 以 使 
每 个 基 类 的 构造 器 都 能 得 到 调用 。 这 样 做 是 有 意义 的 ， 因 为 构造 器 具有 一 项 特殊 任务 : 检查 对 
象 是 否 被 正确 地 构造 。 导 出 类 只 能 访问 它 自己 的 成 员 ， 不 能 访问 基 类 中 的 成 员 ( 基 类 成 员 通常 
是 private 类 型 )。 只 有 基 类 的 构造 器 才 具 有 恰当 的 知识 和 权限 来 对 自己 的 元 素 进行 初始 化 。 因 此 ， 
必须 令 所 有 构造 器 都 得 到 调用 ， 否 则 就 不 可 能 正确 构造 完整 对 象 。 这 正 是 编译 器 为 什么 要 强制 
每 个 导出 类 部 分 都 必须 调用 构造 器 的 原因 。 在 导出 类 的 构造 器 主体 中 ， 如 果 没 有 明确 指定 调用 
某 个 基 类 构造 器 ， 它 就 会 “默默 ”地 调用 默认 构造 器 。 如 果 不 存在 默认 构造 器 ， 编 译 器 就 会 报 
错 ( 若 某 个 类 没有 构造 器 ， 编 译 器 会 自动 合成 出 一 个 默认 构造 器 )。 

让 我 们 来 看 下 面 这 个 例子 ， 它 展示 组 合 、 继 承 以 及 多 态 在 构建 顺序 上 的 作用 : 


/1: polymorphism/Sandwich. java ‘ 
// Order of constructor calls. 

package polymorphism; 

import static net.mindview.util.Print.*; 


class Heal { 
Meal() { print("Meat()"); } 
293 } 
class Bread { 
Bread() { print("Bread()"); } 
} 














class Cheese { 
Cheese() { print("Cheese()"): } 


| class Lettuce { 
Lettuce() { print("Lettuce()"); } 


+ class Lunch extends Meal { 
Lunch() { print(*Lunch()"); } 
$ 


class PortableLunch extends Lunch { 
PortableLunch() { print("PortableLunch()");} 
} 


public class Sandwich extends PortableLunch { 
Private Bread b = new Bread(); 
private Cheese c = new Cheese(); 
private Lettuce 1 = new Lettuce(): 
public Sandwich() { print("Sandwich()"); } 
public static void main(String() args) { 
new Sandwich() ; 


} 

/* Output: 
Meal) 
Lunch() 


PortableLunch() 
Bread() 
Cheese() 
Lettuce() 
Sandwich() 
Whim 


在 这 个 例子 中 ， 用 其 他 类 创建 了 一 个 复杂 的 类 ， 而 且 每 个 类 都 有 一 个 声明 它 自己 的 构造 器 。 

其 中 最 重要 的 类 是 Sandwich， 它 反映 了 三 层 继承 ( 若 将 自 Object 的 隐 含 继承 也 算 在 内 ， 就 是 四 

E) 以 及 三 个 成 员 对 象 。 当 在 main0 里 创建 一 个 Sandwich 对 象 后 ， 就 可 以 看 到 输出 结果 。 这 也 
| 294) ”表明 了 这 一 复杂 对 象 调用 构造 器 要 遵照 下 面 的 顺序 : 
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D 调用 基 类 构造 器 。 这 个 步骤 会 不 断 地 反复 递归 下 去 ， 首 先是 构造 这 种 层次 结构 的 根 ， 然 
后 是 下 一 层 导出 类 ， 等 等 ， 直 到 最 低层 的 导出 类 。 

2) 按 声明 顺序 调用 成 员 的 初始 化 方法 。 

3) 调用 导出 类 构造 器 的 主体 。 

构造 器 的 调用 顺序 是 很 重要 的 。 当 进行 继承 时 ， 我 们 已 经 知道 基 类 的 一 切 ， 并 且 可 以 访问 
基 类 中 任何 声明 为 public 和 protected 的 成 员 。 这 意味 着 在 导出 类 中 ， 必 须 假定 基 类 的 所 有 成 员 
都 是 有 效 的 。 一 种 标准 方法 是 ， 构 造 动 作 一 经 发 生 ， 那 么 对 象 所 有 部 分 的 全 体 成 员 都 会 得 到 构 
建 。 然 而 ， 在 构造 器 内 部 ， 我 们 必须 确保 所 要 使 用 的 成 员 都 已 经 构建 完毕 。 为 确保 这 一 目的 ， 
唯一 的 办 法 就 是 首先 调用 基 类 构造 器 。 那 么 在 进入 导出 类 构造 器 时 ， 在 基 类 中 可 供 我 们 访问 的 
成 员 都 已 得 到 初始 化 。 此 外 ， 知 道 构 造 器 中 的 所 有 成 员 都 有 效 也 是 因为 ， 当 成 员 对 象 在 类 内 进 
行 定义 的 时 候 (比如 上 例 中 的 b、c 和 1) ， 只 要 有 可 能 ， 就 应 该 对 它们 进行 初始 化 (也 就 是 说 ， 
通过 组 合 方法 将 对 象 置 于 类 内 )。 若 遵循 这 一 规则 ， 那 么 就 能 保证 所 有 基 类 成 员 以 及 当前 对 象 的 
成 员 对 象 都 被 初始 化 了 。 但 遗憾 的 是 ， 这 种 做 法 并 不 适用 于 所 有 情况 ， 这 一 点 我 们 会 在 下 一 节 
中 看 到 。 

45111; (1) 向 Sandwichjava 中 添加 Pickle 类 。 
8.3.2 继承 与 清理 

通过 组 合 和 继承 方法 来 创建 新 类 时 ， 永 远 不 必 担 心 对 象 的 清理 问题 ， 子 对 象 通常 都 会 留 给 
垃圾 回收 器 进行 处 理 。 如 果 确 实 遇 到 清理 的 问题 ， 那 么 必须 用 心 为 新 类 创建 dispose0 方 法 (在 
这 里 我 选用 此 名 称 ， 读 者 可 以 提出 更 好 的 )。 并 且 由 于 继承 的 缘故 ， 如 果 我 们 有 其 他 作为 垃圾 回 
收 一 部 分 的 特殊 清理 动作 ， 就 必须 在 导出 类 中 覆盖 dispose0 方 法 。 当 履 盖 被 继承 类 的 dispose() 方 
法 时 ， 务 必 记 住 调用 基 类 版 本 dispose0 方 法 ， 否 则 ， 基 类 的 清理 动作 就 不 会 发 生 。 下 例 就 证 明 
了 这 一 点 : 


1/: polymorphism/Frog.java 
// Cleanup and inheritance. 

package polymorphism; 

import static net.mindview.util.Print.*; 


class Characteristic { 
private String s; 
Characteristic(String s) { 
this.s = s; 
print("Creating Characteristic ”+ s); 


} 
protected void dispose() { 
print("disposing Characteristic ”+ s); 


$ 


class Description { 
private String s; 
Description(String s) { 
this.s = s; 
print("Creating Description "+ s); 


protected void dispose() { 
print("disposing Description ”+ s): 


} 
class LivingCreature { 


private Characteristic p = 
new Characteristic("is alive"); 
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private Description t = 
new Description("Basic Living Creature"); 

LivingCreature() { 
print(“LivingCreature()"); 

} R 


protected void dispose() { 
print("LivingCreature dispose"); 
t.dispose(); 
p.dispose(); 
} 
} 


class Animal extends LivingCreature { 

private Characteristic p = 

new Characteristic("has heart"); 
private Description t = 

new Description("Animal not Vegetable"); 
Animal() { print("Animat()"); } 
protected void dispose() { 

print("Animal dispose”); 

t.dispose(); 

p.dispose(); 

super .dispose() ; 





} 
} 


class Amphibian extends Animal { 
private Characteristic p = 
new Characteristic("can live in water"); 
private Description t = 
new Description("Both water and land"); 
Amphibian() { 
print("Amphibian()"); 
} ° 
protected void dispose() { 
print("Amphibian dispose"); 
t.dispose(); 
p-dispose(); 
super .dispose(); 
} 


public class Frog extends Amphibian ( 

private Characteristic p = new Characteristic("Croaks"); 
private Description t = new Description("Eats Bugs"): 
public Frog() { print("Frog()"); } 
protected void dispose() ( 

print ("Frog dispose"); 

t.dispose(); 

p.dispose(); 

super .dispose(); 
} 
public static void main(String[] args) { 





Frog frog = new Frog(); 
print("Bye!"); 
frog.dispose(); 
》 
} /* Output: 


Creating Characteristic is alive 
Creating Description Basic Living Creature 
LivingCreature() 


“Creating Characteristic has heart 


Creating Description Animal not Vegetable 
Animal () 

Creating Characteristic can live in water 
Creating Description Both water and land 
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Amphibian () 

Creating Characteristic Croaks 

Creating Description Eats Bugs 

Frog() 

Bye! 

Frog dispose 

disposing Description Eats Bugs 

disposing Characteristic Croaks 

Amphibian dispose 

disposing Description Both water and land 
disposing Characteristic can live in water 
Animal dispose 

disposing Description Animal not Vegetable 
disposing Characteristic has heart 
LivingCreature dispose 

disposing Description Basic Living Creature 
disposing Characteristic is alive 

Wha 


层次 结构 中 的 每 个 类 都 包含 Characteristic 和 Deseription 这 两 种 类 型 的 成 员 对 象 ， 并 且 它 们 
也 必须 被 销毁 。 所 以 万 一 某 个 子 对 象 要 依赖 于 其 他 对 象 ， 销 色 的 顺序 应 该 和 初始 化 顺序 相反 。 
对 于 字段 ， 则 意味 着 与 声明 的 顺序 相反 (因为 字段 的 初始 化 是 按照 声明 的 顺序 进行 的 )。 对 于 基 
类 (遵循 C++ 中 析 构 函数 的 形式 )， 应 该 首先 对 其 导出 类 进行 清理 ， 然 后 才 是 基 类 。 这 是 因为 导 
出 类 的 清理 可 能 会 调用 基 类 中 的 某 些 方法 ， 所 以 需要 使 基 类 中 的 构件 仍 起 作用 而 不 应 过 早 地 销 
毁 它们 。 从 输出 结果 可 以 看 到 ，Frog 对 象 的 所 有 部 分 都 是 按照 创建 的 逆序 进行 销毁 的 。 

在 这 个 例子 中 可 以 看 到 ， 尽 管 通常 不 必 执 行 清理 工作 ， 但 是 一 旦 选择 要 执行 ， 就 必须 谨慎 
和 小 心 。 

练习 12: (3) 修改 练习 9， 使 其 能 够 演示 基 类 和 导出 类 的 初始 化 顺序 。 然 后 向 基 类 和 导出 类 
中 添加 成 员 对 象 ， 并 说 明 构 建 期 间 初始 化 发 生 的 顺序 。 

在 上 面 的 示例 中 还 应 该 注意 到 ，Frog 对 象 拥有 其 自己 的 成 员 对 象 。Frog 对 象 创建 了 它 自己 
的 成 员 对 象 ， 并 且 知 道 它们 应 该 存活 多 久 (只 要 Frog 存 活着 ) ， 因 此 Frog 对 象 知道 何 时 调用 
dispose0 去 释放 其 成 员 对 象 。 然 而 ， 如 果 这 些 成 员 对 象 中 存在 于 其 他 一 个 或 多 个 对 象 共享 的 情 
况 ， 问 题 就 变 得 更 加 复杂 了 ， 你 就 不 能 简单 地 假设 你 可 以 调用 dispose0 了 。 在 这 种 情况 下 ， 也 
许 就 必需 使 用 引用 计数 来 跟踪 仍旧 访问 着 共享 对 象 的 对 象 数量 了 。 下 面 是 相关 的 代码 ; 


1/: polymorphism/ReferenceCounting. java 
// Cleaning up shared member objects. 
import static net.mindview.util.Print.*; 


class Shared { 
‘private int refcount = 0; 
private static long counter = @; 
private final long id = counter++; 
public Shared() { 
print("Creating * + this): 


} 
public void addRef() { refcount++; } 
protected void dispose() { 
if(--refcount == @) 
print ("Disposing " + this): 


} 
public String toString() { return "Shared "+ id: } 
} 


class Composing { 
private Shared shared; 
private static long counter = @; 
private final long id = counter++; 
public Composing(Shared shared) { 
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print("Creating ”+ this): 
this.shared = shared: 
this. shared. addRef() : 
} 
protected void dispose() { 
print("disposing " + this); 
shared.dispose(): 
} 
public String toString() { return "Composing " + id; } 
} 


Public class ReferenceCounting { 
public static void main(Stringl] args) { 

Shared shared = new Shared(); 

Composingl] composing = { new Composing(shared), 
new Composing(shared), new Composing(shared), 
new Composing(shared), new Composing(shared) }; 

for(Composing < : composing) 
¢.dispose(); 


} ). Output: 

Creating Shared 9 

Creating Composing 6 

Creating Composing 1 

Creating Composing 2 

Creating Composing 3 

Creating Composing 4 

disposing Composing @ 

disposing Composing 1 

disposing Composing 2 

disposing Composing 3 

disposing Composing 4 

Disposing Shared © 

AAA 

static long counter 跟 踪 所 创建 的 Shared 的 实例 的 数量 ， 还 可 以 为 id 提供 数值 。counter 的 类 
型 是 long 而 不 是 int， 这 样 可 以 防止 游 出 (这 只 是 一 个 良好 实践 ， 对 于 本 书 中 的 所 有 示例 ， 这 种 
计数 器 不 可 能 发 生 溢出 )。id 是 final 的 ， 因 为 我 们 不 希望 它 的 值 在 对 象 生命 周期 中 被 改变 。 

在 将 一 个 共享 对 象 附着 到 类 上 时 ,必须 记 住 调用 addRef0, 但 是 dispose0 方 法 将 跟踪 引用 数 ， 
并 决定 何 时 执行 清理 。 使 用 这 种 技巧 需要 加 倍 地 细心 ， 但 是 如 果 你 正在 共享 需要 清理 的 对 象 ， 
那么 你 就 没有 太 多 的 选择 余地 了 。 

练习 13，(3) 在 ReferenceCounting.java 中 添加 一 个 finalize0 方 法 ， 用 来 校 验 终止 条 件 (查看 
第 5 章 )。 i 

练习 14: (4) 修改 练习 12， 使 得 其 某 个 成 员 对 象 变 为 具有 引用 计数 的 共享 对 象 ， 并 证 明 它 可 
以 正确 运行 。 
8.3.3 构造 器 内 部 的 多 态 方法 的 行为 

构造 器 调用 的 层次 结构 带 来 了 一 个 有 趣 的 两 难 问题 。 如 果 在 一 个 构造 器 的 内 部 调用 正在 构 
造 的 对 象 的 某 个 动态 绑 定 方法 ， 那 会 发 生 什么 情况 呢 ? 

在 一 般 的 方法 内 部 ， 动 态 绑 定 的 调用 是 在 运行 时 才 决 定 的 ， 因 为 对 象 无 法 知道 它 是 属于 方 
法 所 在 的 那个 类 ， 还 是 属于 那个 类 的 导出 类 。 

如 果 要 调用 构造 器 内 部 的 一 个 动态 绑 定 方法 ， 就 要 用 到 那个 方法 的 被 覆盖 后 的 定义 。 然 而 ， 
这 个 调用 的 效果 可 能 相当 难于 预料 ， 因 为 被 覆盖 的 方法 在 对 象 被 完全 构造 之 前 就 会 被 调用 。 这 可 
能 会 造成 一 些 难于 发 现 的 隐藏 错误 。 

从 概念 上 讲 ， 构 造 器 的 工作 实际 上 是 创建 对 象 (这 并 非 是 一 件 平常 的 工作 )。 在 任何 构造 器 
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内 部 ， 整 个 对 象 可 能 只 是 部 分 形成 一 -我 们 只 知道 基 类 对 象 已 经 进行 初始 化 。 如 果 构 造 器 只 是 
在 构建 对 象 过 程 中 的 一 个 步 又， 并 且 该 对 象 所 属 的 类 是 从 这 个 构造 器 所 属 的 类 导出 的 ， 那 么 导 
出 部 分 在 当前 构造 器 正在 被 调用 的 时 刻 仍旧 是 没有 被 初始 化 的 。 然 而 ， 一 个 动态 绑 定 的 方法 调 
用 却 会 向 外 深入 到 继承 层次 结构 内 部 ， 它 可 以 调用 导出 类 里 的 方法 。 如 果 我 们 是 在 构造 器 内 部 
这 样 做 ， 那 么 就 可 能 会 调用 某 个 方法 ， 而 这 个 方法 所 操纵 的 成 员 可 能 还 未 进行 初始 化 一 这 肯 
定 会 招致 灾难 。 

通过 下 面 这 个 例子 ， 我 们 会 看 到 问题 所 在 : 


/1/: polymorphism/PolyConstructors. java 

// Constructors and polymorphism 

// don't produce what you might expect. 
import static net.mindview.util.Print.*; 


class Glyph { 
void draw() { print("Glyph.draw()"); } 
Glyph() { 
Print("Glyph() before draw()"); 
draw(); 
print("Glyph() after draw()"); 


} 


class RoundGlyph extends Glyph { 
private int radius = 1; 
RoundGlyph(int r) { 
radius = r; 
print("RoundGlyph.RoundGlyph(), radius = * + radius); 


} 
void draw() { 

print("RoundGlyph.draw(), radius = ”+ radius); 
} 


public class PolyConstructors { 
Public static void main(String{] args) { 
new RoundGlyph(5) ; 
$. 
} /* Output: 
Glyph() before draw() 
RoundGlyph.draw(), radius = @ 
Glyph() after draw() 
RoundGlyph.RoundGlyph(), radius = 5 
Whim 


GIyph.draw(0 方 法 设计 为 将 要 被 覆盖 ， 这 种 覆盖 是 在 RoundGlyph 中 发 生 的 。 但 是 Glyph 构 
造 器 会 调用 这 个 方法 ， 结 果 导致 了 对 RoundGIlyph.draw0 的 调用 ， 这 看 起 来 似乎 是 我 们 的 目的 。 
但 是 如 果 看 到 输出 结果 ， 我 们 会 发 现 当 Glyph 的 构造 器 调用 draw0 方 法 时 ，radius 不 是 默认 初始 
值 1， 而 是 0。 这 可 能 导致 在 屏幕 上 只 画 了 一 个 点 ， 或 者 根本 什么 东西 都 没有 ， 我 们 只 能 干 瞪眼 ， 
并 试图 找 出 程序 无 法 运转 的 原因 所 在 。 

前 一 节 讲 述 的 初始 化 顺序 并 不 十 分 完整 ， 而 这 正 是 解决 这 一 谜 题 的 关键 所 在 。 初 始 化 的 实 
际 过 程 是 : 

1) 在 其 他 任何 事物 发 生 之 前 ， 将 分 配给 对 象 的 存储 空间 初始 化 成 二 进 制 的 零 。 

2) 如 前 所 述 那样 调用 基 类 构造 器 。 此 时 ， 调 用 被 覆盖 后 的 draw() 方 法 (要 在 调用 
RoundGIlyph 构 造 器 之 前 调用 ) ， 由 于 步骤 1 的 缘故 ， 我 们 此 时 会 发 现 radius 的 值 为 0。 

3) 按照 声明 的 顺序 调用 成 员 的 初始 化 方法 。 
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4) 调用 导出 类 的 构造 器 主体 。 

这 样 做 有 一 个 优点 ， 那 就 是 所 有 东西 都 至 少 初始 化 成 零 (或 者 是 某 些 特殊 数据 类 型 中 与 
“ 零 ” 等 价 的 值 )， 而 不 是 仅仅 留 作 垃圾 。 其 中 包括 通过 “组 合 ”而 戏 入 一 个 类 内 部 的 对 象 引用 ， 
其 值 是 null。 所 以 如 果 忘 记 为 该 引 用 进行 初始 化 ， 就 会 在 运行 时 出 现 异常 。 查 看 输出 结果 时 ， 会 
发 现 其 他 所 有 东西 的 值 都 会 是 零 ， 这 通常 也 正 是 发 现 问题 的 证 据 。 

另 一 方面 ， 我 们 应 该 对 这 个 程序 的 结果 相当 震惊 。 在 逻辑 方面 ， 我 们 做 的 已 经 十 分 完美 ， 
而 它 的 行为 却 不 可 思议 地 错 了 ， 并 且 编 译 器 也 没有 报错 。( 在 这 种 情况 下 ，C++ 语 言 会 产生 更 合 
理 的 行为 。) 诸如 此 类 的 错误 会 很 容易 被 人 忽略 ,而且 要 花 很 长 的 时 间 才 能 发 现 。 

因此 ， 编 写 构造 器 时 有 一 条 有 效 的 准则 :“ 用 尽 可 能 简单 的 方法 使 对 象 进入 正常 状态 ， 如 果 
可 以 的 话 ， 避 免 调用 其 他 方法 "。 在 构造 器 内 唯一 能 够 安全 调用 的 那些 方法 是 基 类 中 的 final 方 法 
(也 适用 于 private 方 法 ， 它 们 自动 属于 final 方 法 )。 这 些 方法 不 能 被 覆盖 ， 因 此 也 就 不 会 出 现 上 
述 令 人 惊讶 的 问题 。 你 可 能 无 法 总 是 能 够 遵循 这 条 准则 ， 但 是 应 该 朝 着 它 努 力 。 

练习 15，(2) 在 PolyConstructors.java 中 添加 一 个 RectangularGlyph， 并 证 明 会 出 现 本 节 所 
描述 的 问题 。 


8.4 协 变 返 回 类 型 


Java SE5 中 添加 了 协 变 返 回 类 型 ， 它 表示 在 导出 类 中 的 被 覆盖 方法 可 以 返回 基 类 方法 的 返回 
类 型 的 某 种 导出 类 型 ; 


711; polymorphism/CovariantReturn. java 


class Grain { 
public String toString() { return "Grain"; } 


} 


class Wheat extends Grain { 
public String toString() { return “Wheat; } 
} 


class Mill { 
Grain process() { return new Grain(); } 


} 


class WheatMill extends Mill { 
Wheat process() { return new Wheat(); } 


public class CovariantReturn { 
public static void main(String{] args) { 
Mill m = new MULO); 
Grain g = m.process(); 
System. out.printin(g): 
m = new WheatMi1l(); 
g = m.process(); 
System.out.printin(g); 
} 
} /* Output: 
Grain 
Wheat 
#111:~ 


Java SE5 与 Java 较 早 版 本 之 间 的 主要 差异 就 是 较 早 的 版 本 将 强制 process0 的 覆盖 版 本 必须 返 
回 Grain， 而 不 能 返回 Wheat， 尽 管 Wheat 是 从 Grain 导 出 的 ， 因 而 也 应 该 是 一 种 合法 的 返回 类 
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型 。 协 变 返 回 类 型 允许 返回 更 具体 的 Wheat 类 型 。 
8.5 用 继承 进行 设计 

学 习 了 多 态 之 后 ， 看 起 来 似乎 所 有 东西 都 可 以 被 继承 ， 因 为 多 态 是 一 种 如 此 巧妙 的 工具 。 
事实 上 ， 当 我 们 使 用 现成 的 类 来 建立 新 类 时 ， 如 果 首先 考虑 使 用 继承 技术 ， 反 倒 会 加 重 我 们 的 
设计 负担 ， 使 事情 变 得 不 必要 地 复杂 起 来 。 

更 好 的 方式 是 首先 选择 “组 合 "， 尤 其 是 不 能 十 分 确定 应 该 使 用 哪 一 种 方式 时 。 组 合 不 会 强 
制 我 们 的 程序 设计 进入 继承 的 层次 结构 中 。 而 且 ， 组 合 更 加 灵活 ， 因 为 它 可 以 动态 选择 类 型 
(因此 也 就 选择 了 行为 ) ， 相 反 ， 继 承 在 编译 时 就 需要 知道 确切 类 型 。 下 面 举例 说 明 这 一 点 : 


1/: potymorphism/Transmogrify.java 
// Dynamically changing the behavior of an object 
// via composition (the “State” design pattern). 
inport static net.mindview.util.Print.*; 


class Actor { 
public void act() {} 
} 


class HappyActor extends Actor { 
public void act() { print("HappyActor"); } 
} 


class SadActor extends Actor { 
public void act() { print("SadActor"); } 


class Stage { 
private Actor actor = new HappyActor(); 
public void change() { actor = new SadActor(); } 
public void performPlay() { actor.act(); } 

) 


public class Transmogrify { 
public static void main(String(] args) { 
Stage stage = new Stage(); 
stage. performPlay() ; 
stage. change(); 
stage.performPlay() ; 


} h Output: 

HappyActor 

SadActor 

AAA: 

在 这 里 ，Stage 对 象 包含 一 个 对 Actor 的 引用 ， 而 Actor 被 初始 化 为 HappyActor 对 象 。 这 意味 
着 performPiay0 会 产生 某 种 特殊 行为 既然 引用 在 运行 时 可 以 与 另 一 个 不 同 的 对 象 重新 绑 定 起 来 ， 
所 以 SadAetor 对 象 的 引用 可 以 在 actor 中 被 替代 ， 然 后 由 performPlay0 产 生 的 行为 也 随 之 改变 。 
这 样 一 来 ， 我 们 在 运行 期 间 获得 了 动态 灵活 性 (这 也 称 作 状 态 模式 ， 请 参阅 www.MindView.com 
上 的 《Thinking in Patterns (with Java)》)。 与 此 相反 ， 我 们 不 能 在 运行 期 间 决定 继承 不 同 的 对 象 ， 
因为 它 要 求 在 编译 期 间 完全 确定 下 来 。 

一 条 通用 的 准则 是 :“ 用 继承 表达 行为 间 的 差异 ， 并 用 字段 表达 状态 上 的 变化 "。 在 上 述 例 
子 中 ， 两 者 都 用 到 了 : 通过 继承 得 到 了 两 个 不 同 的 类 ， 用 于 表达 aet( 方 法 的 差异 ， 而 Stage 通 过 
运用 组 合 使 自己 的 状态 发 生变 化 。 在 这 种 情况 下 ， 这 种 状态 的 改变 也 就 产生 了 行为 的 改变 。 

练习 16: (3) 遵循 Transmogrifyjava 这 个 例子 ， 创 建 一 个 Starship 类 ， 包 含 一 个 AlertStatus 
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引用 ， 此 引用 可 以 指示 三 种 不 同 的 状态 。 纳 入 一 些 可 以 改变 这 些 状态 的 方法 。 


8.5.1 纯 继承 与 扩展 


采取 “纯粹 ”的 方式 来 创建 继承 层次 结构 似乎 是 最 好 的 方式 。 也 就 是 说 ， 只 有 在 基 类 中 已 


经 建立 的 方法 才 可 以 在 导出 类 中 被 覆盖 ， 如 下 图 所 示 ; 





draw() 
erase() 




















circle | | Square { Trianele | 
draw() | darawO draw() 
erase() erase() erase() 














这 被 称 作 是 纯粹 的 “is-a”( 是 一 种 ) 关系 ， 因 为 一 个 类 的 接口 已 经 确定 了 它 应 该 是 什么 。 
继承 可 以 确保 所 有 的 导出 类 具有 基 类 的 接口 ， 且 绝对 不 会 少 。 按 上 图 那么 做 ， 导 出 类 也 将 具有 


和 基 类 一 样 的 接口 。 


也 可 以 认为 这 是 一 种 纯 普 代 ， 因 为 导出 类 可 以 完全 代替 基 类 ， 而 在 使 用 它们 时 ， 完 全 不 需 


要 知道 关于 子 类 的 任何 额外 信息 : 








也 就 是 说 ， 基 类 可 以 接收 发 送 给 导出 类 的 任何 消息 ， 因 为 二 
者 有 着 完全 相同 的 接口 。 我 们 只 需 从 导出 类 向 上 转型 ， 永 远 不 需 
知道 正在 处 理 的 对 象 的 确切 类 型 。 所 有 这 一 切 ， 都 是 通过 多 态 来 
处 理 的 (如 右 图 所 示 )。 

按 这 种 方式 考虑 ， 似 乎 只 有 纯粹 的 is-a 关 系 才 是 唯一 明智 的 
做 法 ， 而 所 有 其 他 的 设计 都 只 会 导致 混乱 和 注定 会 失败 。 这 其 
实 也 是 一 个 陷阱 ， 因 为 只 要 开始 考虑 ， 就 会 转向 ， 并 发 现 扩展 
接口 (遗憾 的 是 ，extends 关 键 字 似乎 在 候 巴 我 们 这 样 做 ) 才 是 
解决 特定 问题 的 完美 方案 。 这 可 以 称 为 “is-like-a”( 像 一 个 ) 关 
系 ， 因 为 导出 类 就 像 是 一 个 基 类 一 一 它 有 着 相同 的 基本 接口 , 但 
是 它 还 具有 由 额外 方法 实现 的 其 他 特性 。 


bape 对话 | Cirle, Square, Lines 
“Is-a” 关系 新 的 Shape 类 型 

















Useful 
void f() 假定 这 代表 
void g() } 一 个 大 的 接口 
Moreuseful | “Is-like-a” 关 系 


| voia t0 


void 9() 
void u0) 


| void vO 





扩展 接口 
void w() 小 


虽然 这 是 一 种 有 用 且 明 智 的 方法 (依赖 于 具体 情况 )， 但 是 它 也 有 缺点 。 导 出 类 中 接口 的 扩 
展 部 分 不 能 被 基 类 访问 ， 因 此 ， 一 旦 我 们 向 上 转型 ， 就 不 能 调用 那些 新 方法 : 








与 Useful rr 
对 象 对 话 








多 & 167 





在 这 种 情况 下 ， 如 果 我 们 不 进行 向 上 转型 ， 这 样 的 问题 也 就 不 会 出 现 。 但 是 通常 情况 下 ， 
我 们 需要 重新 查 明 对 象 的 确切 类 型 ， 以 便 能 够 访问 该 类 型 所 扩充 的 方法 。 下 一 节 将 说 明 如 何 做 
到 这 一 点 。 

8.5.2 向 下 转型 与 运行 时 类 型 识别 

由 于 向 上 转型 (在 继承 层次 中 向 上 移动 ) 会 丢失 具体 的 类 型 信息 ， 所 以 我 们 就 想 ， 通 过 向 
下 转型 一 也 就 是 在 继承 层次 中 向 下 移动 一 应 该 能 够 获取 类 型 信息 。 然 而 ， 我 们 知道 向 上 转型 
是 安全 的 ， 因 为 基 类 不 会 具有 大 于 导出 类 的 接口 。 因 此 ， 我 们 通过 基 类 接口 发 送 的 消息 保证 都 
能 被 接受 。 但 是 对 于 向 下 转型 ， 例 如 ， 我 们 无 法 知道 一 个 “几何 形状 ” 它 确实 就 是 一 个 “ 圆 ”， 
它 可 以 是 一 个 三 角形 、 正 方形 或 其 他 一 些 类 型 。 

要 解决 这 个 问题 ， 必 须 有 某 种 方法 来 确保 向 下 转型 的 正确 性 ， 使 我 们 不 致 于 贸然 转型 到 一 
种 错误 类 型 ， 进 而 发 出 该 对 象 无 法 接受 的 消息 。 这 样 做 是 极其 不 安全 的 。 

在 某 些 程序 设计 语言 (如 C++) 中 ， 我 们 必须 执行 一 个 特殊 的 操作 来 获得 安全 的 向 下 转型 。 
但 是 在 Java 语 言 中 ， 所 有 转型 都 会 得 到 检查 ! 所 以 即使 我 们 只 是 进行 一 次 普通 的 加 括 弧 形式 的 
类 型 转换 ， 在 进入 运行 期 时 仍然 会 对 其 进行 检查 ， 以 便 保证 它 的 确 是 我 们 希望 的 那 种 类 型 。 如 
果 不 是 ， 就 会 返回 一 个 ClassCastException (类 转型 异常 )。 这 种 在 运行 期 间 对 类 型 进行 检查 的 
行为 称 作 “ 运 行 时 类 型 识别 ”(RTTI) 。 下 面 的 例子 说 明 RTTI 的 行为 : 

/1/: polymorphism/RTTI. java 


” /1/ Downcasting & Runtime type information (RTTI). 
11 (Throwsexcept ion} 


class Useful { 
public void fO {} 
public void g() {} 
} 


class MoreUseful extends Useful { 
public void f() {} 
public void g() {} 
public void u() {} 
public void vO {} 
public void w() {} 
} 
public class RTTI { 
public static void main(String{] args) { 
Useful(] x = { 
new Useful(), 
new MoreUseful() 


x0] .FO; 

x(1].80; 

// Compile time: method not found in Useful: 
WE x[1] uO: 

((MoreUseful)x{1}).u(); // Downcast/RTTI 
((MoreUseful)x{@]).u(); // Exception thrown 


} 
Uh 
正如 前 一 个 示意 图 中 所 示 ，MoreUseful (更 有 用 的 ) HOP RT Useful (有 用 的 ) 接口 但 
是 由 于 它 是 继承 而 来 的 ， 所 以 它 也 可 以 向 上 转型 到 Useful 类 型 。 我 们 在 main0 方 法 中 对 数组 x 进 
行 初始 化 时 可 以 看 到 这 种 情况 的 发 生 。 既 然 数组 中 的 两 个 对 象 都 属于 Useful 类 ， 所 以 我 们 可 以 调 
用 fO 和 g0 这 两 个 方法 。 如 果 我 们 试图 调用 u0O 方 法 〈 它 只 存在 于 MoreUseful) ， 就 会 返回 一 条 编 
译 时 出 错 消息 。 
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如 果 想 访问 MoreUseful 对 象 的 扩展 接口 ， 就 可 以 尝试 进行 向 下 转型 。 如 果 所 转 类 型 是 正确 
的 类 型 ， 那 么 转型 成 功 ， 否则， 就 会 返回 一 个 ClassCastException 异 常 。 我 们 不 必 为 这 个 异常 编 
写 任何 特殊 的 代码 ， 因 为 它 指出 的 是 程序 员 在 程序 中 任何 地 方 都 可 能 会 犯 的 错误 。{Throws- 
Exception} 注 释 标签 告知 本 书 的 构建 系统 : 在 运行 该 程序 时 ， 预 期 抛 出 一 个 异常 。 

RTTI 的 内 容 不 仅仅 包括 转型 处 理 。 例 如 它 还 提供 一 种 方法 ， 使 你 可 以 在 试图 向 下 转型 之 前 ， 
查看 你 所 要 处 理 的 类 型 。 第 14 章 专门 讨论 Java 运 行 时 类 型 识别 的 所 有 不 同方 面 。 

练习 17: (2) 使 用 练习 1 中 的 Cycle 的 层次 结构 ， 在 Unicycle 和 Bicycle 中 添加 balance0 方 法 ， 
而 Tricycle 中 不 添加 。 创 建 所 有 这 三 种 类 型 的 实例 ， 并 将 它们 向 上 转型 为 Cycle 数组 。 在 该 数组 
的 每 一 个 元 素 上 都 尝试 调用 balance0 ， 并 观察 结果 。 然 后 将 它们 向 下 转型 ， 再 次 调用 balance0 ， 
并 观察 将 所 产生 什么 。 


8.6 总 结 


多 态 意味 着 “不 同 的 形式 "。 在 面向 对 象 的 程序 设计 中 ， 我 们 持 有 从 基 类 继承 而 来 的 相同 接 
口 ， 以 及 使 用 该 接口 的 不 同形 式 : 不 同 版 本 的 动态 绑 定 方法 。 

在 本 章 中 我 们 已 经 知道 ， 如 果 不 运用 数据 抽象 和 继承 ， 就 不 可 能 理解 或 者 甚至 不 可 能 创建 
多 态 的 例子 。 多 态 是 一 种 不 能 单独 来 看 待 的 特性 (例如 ， 像 switeh 语 句 是 可 以 的 )， 相 反 它 只 能 
作为 类 关系 “全 景 ”中 的 一 部 分 ， 与 其 他 特性 协同 工作 。 

为 了 在 自己 的 程序 中 有 效 地 运用 多 态 乃至 面向 对 象 的 技术 ， 必 须 扩展 自己 的 编程 视野 ， 使 
其 不 仅 包 括 个 别 类 的 成 员 和 消息 ， 而 且 还 要 包括 类 与 类 之 间 的 共同 特性 以 及 它们 之 间 的 关系 。 
尽管 这 需要 极 大 的 努力 ， 但 是 这 样 做 是 非常 值得 的 ， 因 为 它 可 以 带 来 很 多 成 效 : 更 快 的 程序 开 
发 过 程 、 更 好 的 代码 组 织 、 更 好 扩展 的 程序 以 及 更 容易 的 代码 维护 等 。 

所 选 习题 的 答案 都 可 以 在 名 为 The Thinking in Java Annotated Solution Guide 的 电子 文档 中 找 
到 ， 读 者 可 以 从 www.MindView.net 处 购买 此 文档 。 





第 9 章 接 D 


接口 和 内 部 类 为 我 们 提供 了 一 种 将 接口 与 实现 分 离 的 更 加 结构 化 的 方法 。 

这 种 机 制 在 编程 语言 中 并 不 通用 。 例 如 ，C++ 对 这 些 概念 只 有 间接 的 支持 。 在 Java 中 存在 语 
言 关键 字 这 个 事实 表明 人 们 认为 这 些 思想 是 很 重要 的 ， 以 至 于 要 提供 对 它们 的 直接 支持 。 

首先 ， 我 们 将 学 习 抽象 类 ， 它 是 普通 的 类 与 接口 之 间 的 一 种 中 庸 之 道 。 尽 管 在 构建 具有 茶 
些 未 实现 方法 的 类 时 ， 你 的 第 一 想法 可 能 是 创建 接口 ， 但 是 抽象 类 仍旧 是 用 于 此 目的 的 一 种 重 
要 而 必须 的 工具 。 因 为 你 不 可 能 总 是 使 用 纯 接口 。 


9.1 抽象 类 和 抽象 方法 


在 第 8 章 所 有 “乐器 ”的 例子 中 ， 基 类 Instrument 中 的 方法 往往 是 “ 哑 ”(dummy) 方法 。 
车 要 调用 这 些 方 法 ， 就 会 出 现 一 些 错误 。 这 是 因为 Instrument 类 的 目的 是 为 它 的 所 有 导出 类 创 
建 一 个 通用 接口 。 

在 那些 示例 中 ， 建 立 这 个 通用 接口 的 唯一 理由 是 ， 不 同 的 子 类 可 以 用 不 同 的 方式 表示 此 接 
口 。 通 用 接口 建立 起 一 种 基本 形式 ， 以 此 表示 所 有 导出 类 的 共同 部 分 。 另 一 种 说 法 是 将 
Instrument 类 称 作 抽象 基 类 ， 或 简称 抽象 类 。 

如 果 我 们 只 有 一 个 像 Instrument 这 样 的 抽象 类 ， 那 么 该 类 的 对 象 几乎 没有 任何 意义 。 我 们 
创建 抽象 类 是 希望 通过 这 个 通用 接口 操纵 一 系列 类 。 因 此 ，Instrument 只 是 表示 了 一 个 接口 ， 
没有 具体 的 实现 内 容 ， 因 此 ， 创 建 一 个 Instrument 对 象 没有 什么 意义 ， 并 且 我 们 可 能 还 想 阻止 
使 用 者 这 样 做 。 通 过 让 Instrument 中 的 所 有 方 都 产生 错误 ， 就 可 以 实现 这 个 目的 。 但 是 这 样 做 
会 将 错误 信息 延迟 到 运行 时 才 获 得 ， 并 且 需 要 在 客户 端 进行 可 靠 、 详 尽 的 测试 。 所 以 最 好 是 在 
编译 时 捕获 这 些 问题 。 

为 此 , Java 提 供 一 个 叫做 抽象 方法 。 的 机 制 , 这 种 方法 是 不 完整 的 ， 仅 有 声明 而 没有 方法 体 。 
下 面 是 抽象 方法 声明 所 采用 的 语法 : 

abstract void t0; 

包含 抽象 方法 的 类 叫做 抽象 类 。 如 果 一 个 类 包含 一 个 或 多 个 抽象 方法 ， 该 类 必须 被 限定 为 
抽象 的 。( 否 则 ， 编 译 器 就 会 报错 .) 

如 果 一 个 抽象 类 不 完整 ， 那 么 当 我 们 试图 产生 该 类 的 对 象 时 ， 编 译 器 会 怎样 处 理 呢 ? 由 于 
为 抽象 类 创建 对 象 是 不 安全 的 ， 所 以 我 们 会 从 编译 器 那里 得 到 一 条 出 错 消息 。 这 样 ， 编 译 器 会 
确保 抽象 类 的 纯粹 性 ， 我 们 不 必 担 心 会 误 用 它 。 

如 果 从 一 个 抽象 类 继承 ， 并 想 创 建 该 新 类 的 对 象 ， 那 么 就 必须 为 基 类 中 的 所 有 抽象 方法 提 
供 方法 定义 。 如 果 不 这 样 做 (可 以 选择 不 做 )， 那 么 导出 类 便 也 是 抽象 类 ， 且 编译 器 将 会 强制 我 
们 用 abstract 关 键 字 来 限定 这 个 类 。 

我 们 也 可 能 会 创建 一 个 没有 任何 抽象 方法 的 抽象 类 。 考 虑 这 种 情况 : 如 果 有 一 个 类 ， 让 其 


O 对 于 C++ 程序 设计 员 来 说 ， 这 相当 于 C++ 语言 中 的 纯 虚 函数 。 


be 
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包含 任何 abstract 方 法 都 显得 没有 实际 意义 ， 而 且 我 们 也 想 要 阻止 产生 这 个 类 的 任何 对 象 ， 那 么 
这 时 这 样 做 就 很 有 用 了 。 

第 8 章 Instrument 类 可 以 很 容易 地 转化 成 abstract 类 。 既 然 使 某 个 类 成 为 抽象 类 并 不 需要 所 
有 的 方法 都 是 抽象 的 ， 所 以 仅 需 将 某 些 方法 声明 为 抽象 的 即 可 。 如 下 图 所 示 : 


abstract Instrument ` | 


abstract void play(); 


String what() { /* ... */ } | 
































abstract void adjust(); | 
extends extends inds 
wna Cg | 
void play() void play() void play() 
String what() | | String what() | | String what() 
void adjust() | | vold adjust) | | vold adjust() 
extgnds edna 
Woodwind [brass 
void play() | void play() 
String what() void adjust() 














下 面 是 修改 过 的 “管弦 乐器 ”的 例子 ， 其 中 采用 了 抽象 类 和 抽象 方法 : 


/1/: interfaces/music4/Music4. java 

// Abstract classes and methods. 

package interfaces.music4; 

import polymorphism.music.Note; 

import static net.mindview.util.Print.*; 


abstract class Instrument { 
private int i; // Storage allocated for each 
public abstract void play(Note n); 
public String what() { return "Instrument"; } 
public abstract void adjust(); 

} 


class Wind extends Instrument { 
public void play(Note n) { 
print("Wind.play() " + n); 
$ 


public String what() { return "Wind"; } 
public void adjust() {} 
} 


class Percussion extends Instrument { 
public void play(Note n) { 
print("Percussion.play() * + n); 


} 
public String what() { return "Percussion"; } 
public void adjust() {} 

} 


class Stringed extends Instrument { 
public void play(Note n) { 
print("Stringed.play() " + n); 
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} 
public String what() { return "Stringed"; } 
public void adjust() {} 

} 


Class Brass extends Wind { 
public void play(Note n) { 
print("Brass.play() ”+ n); 


} 
public void adjust() { print("Brass.adjust()"); } 
} 


class Woodwind extends Wind { 
public void play(Note n) { 
print("Woodwind.play() ”+ n); 


} 
public String what() { return "Woodwind"; } 


public class Husic4 { 
// Doesn't care about type, so new types 
// added to the system still work right: 
static void tune(Instrument i) { 
ei 
1.play (Note. MIDDLE_C) ; 


static void tuneAll(Instrument[{] e) { 
for (Instrument i : e) 
tune(i); 
} 
public static void main(String(] args) { 
// Upcasting during addition to the array: 
Instrument[] orchestra = { 
new Wind(), 
new Percussion(), 
new Stringed(), 
new Brass(), 
new Woodwind() 


yi 
tuneAll (orchestra) ; 


} 
} /* Output: 
Wind.play() MIDDLE_C 
Percussion.play() MIDDLE_C 
Stringed. play() MIDDLE_C 
Brass.play() MIDDLE_C 
Woodwind.play() MIDDLE_C 
Whim 


我 们 可 以 看 出 ， 除 了 基 类 ， 实 际 上 并 没有 什么 改变 。 

创建 抽象 类 和 抽象 方法 非常 有 用 ， 因 为 它们 可 以 使 类 的 抽象 性 明确 起 来 ， 并 告诉 用 户 和 编 
译 器 打算 怎样 来 使 用 它们 。 抽 象 类 还 是 很 有 用 的 重 构 工具 ， 因 为 它们 使 得 我 们 可 以 很 容易 地 将 
公共 方法 沿 着 继承 层次 结构 向 上 移动 。 ’ 

练习 1: (1) 修改 第 8 章 练习 9 中 的 Rodent， 使 其 成 为 一 个 抽象 类 。 只 要 有 可 能 ， 就 将 Rodent 
的 方法 声明 为 抽象 方法 。 

练习 2: (1) 创建 一 个 不 包含 任何 抽象 方法 的 抽象 类 ， 并 验证 我 们 不 能 为 该 类 创建 任何 实例 。 

练习 3: (2) 创建 一 个 基 类 ， 让 它 包含 抽象 方法 print0， 并 在 导出 类 中 覆盖 该 方法 。 狂 盖 后 
的 方法 版 本 可 以 打印 导出 类 中 定义 的 某 个 整 型 变量 的 值 。 在 定义 该 变量 之 处 ， 赋予 它 非 零 值 。 
在 基 类 的 构造 器 中 调用 这 个 方法 。 现在， 在 main0 方 法 中 ， 创 建 一 个 导出 类 对 象 ， 然后 调用 它 
的 print0 方 法 。 请 解释 发 生 的 情形 。 
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练习 4: (3) 创建 一 个 不 包含 任何 方法 的 抽象 类 ， 从 它 那 里 导出 一 个 类 ， 并 添加 一 个 方法 。 
创建 一 个 静态 方法 ， 它 可 以 接受 指向 基 类 的 引用 ， 将 其 向 下 转型 到 导出 类 ， 然 后 再 调用 该 静态 
方法 。 在 main0 中 ， 展 现 它 的 运行 情况 。 然 后 ， 为 基 类 中 的 方法 加 上 abstract 声 明 ， 这 样 就 不 再 
需要 进行 向 下 转型 。 


9.2 接口 


interface 关 键 字 使 抽象 的 概念 更 向 前 迈进 了 一 步 。abstract 关 键 字 允许 人 们 在 类 中 创建 一 个 
或 多 个 没有 任何 定义 的 方法 一 一 提供 了 接口 部 分 ， 但 是 没有 提供 任何 相应 的 具体 实现 ， 这 些 实 
现 是 由 此 类 的 继承 者 创建 的 。interface 这 个 关键 字 产 生 一 个 完全 抽象 的 类 ， 它 根本 就 没有 提供 
任何 具体 实现 。 它 允许 创建 者 确定 方法 名 、 参 数列 表 和 返回 类 型 ， 但 是 没有 任何 方法 体 。 接 口 
只 提供 了 形式 ， 而 未 提供 任何 具体 实现 。 

一 个 接口 表示 :“ 所 有 实现 了 该 特定 接口 的 类 看 起 来 都 像 这 样 "。 因 此 ， 任 何 使 用 某 特定 接 
口 的 代码 都 知道 可 以 调用 该 接口 的 哪些 方法 ， 而 且 仅 需 知道 这 些 。 因 此 ， 接 口 被 用 来 建立 类 与 
类 之 间 的 协议 。( 某 些 面向 对 象 编程 语言 使 用 关键 字 protocol 来 完成 这 一 功能 。) 

但 是 ，interface 不 仅仅 是 一 个 极度 抽象 的 类 ， 因 为 它 允 许 人 们 通过 创建 一 个 能 够 被 向 上 转 
型 为 多 种 基 类 的 类 型 ， 来 实现 某 种 类 似 多 重 继 变种 的 特性 。 

要 想 创建 一 个 接口 ， 需 要 用 interface 关 键 字 来 替代 class 关 键 字 。 就 像 类 一 样 ， 可 以 在 interface 
关键 字 前 面 添加 public 关 键 字 (但 仅 限于 该 接口 在 与 其 同名 的 文件 中 被 定义 )。 如 果 不 添加 public 
关键 字 ， 则 它 只 具有 包 访问 权限 ， 这 样 它 就 只 能 在 同一 个 包 内 可 用 。 接 口 也 可 以 包含 域 ,但 是 这 
些 域 隐 式 地 是 static 和 final 的 。 

要 让 一 个 类 遵循 某 个 特定 接口 (或 者 是 一 组 接口 )， 需 要 使 用 implements 关 键 字 ， 它 表示 ;: 
“interface 只 是 它 的 外 貌 ， 但 是 现在 我 要 声明 它 是 如 何 工作 的 。” 除 此 之 外 ， 它 看 起 来 还 很 像 继 
承 。“ 乐 器 ”示例 的 图 说 明了 这 一 点 : 


interface Instrument 


void play(); 
String what(); 
void adjust(); 









































Implements implements implements 
Wind Percussion Stringed 
void play() void play() void play() 
String what() | | String what() | | String what() 
void adjust() void adjust() void adjust() 
extgnds 区 
Woodwind Brass 
void play() void play() 
String what() void adjust() 

















可 以 从 Woodwind 和 Brass 类 中 看 到 ， 一 旦 实现 了 某 个 接口 ， 其 实现 就 变 成 了 一 个 普通 的 类 ， 
就 可 以 按 常规 方式 扩展 它 。 
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可 以 选择 在 接口 中 显 式 地 将 方法 声明 为 public 的 ， 但 即使 你 不 这 么 做 ， 它 们 也 是 public 的 。 
因此 ， 当 要 实现 一 个 接口 时 ， 在 接口 中 被 定义 的 方法 必须 被 定义 为 是 public 的 ， 否则， 它们 将 只 
能 得 到 默认 的 包 访问 权限 ， 这 样 在 方法 被 继承 的 过 程 中 ， 其 可 访问 权限 就 被 降低 了 ， 这 是 Java 
编译 器 所 不 允许 的 。 

读者 可 以 在 修改 过 的 Instrument 的 例子 中 看 到 这 一 点 。 要 注意 的 是 ， 在 接口 中 的 每 一 个 方 
法 确实 都 只 是 一 个 声明 ， 这 是 编译 器 所 允许 的 在 接口 中 唯一 能 够 存在 的 事物 。 此 外 ， 在 
JInstrument 中 没有 任何 方法 被 声明 为 是 public 的 ， 但 是 它们 自动 就 都 是 public 的 : 


/1/: interfaces/musicS/MusicS. java 

// Interfaces. ， 
package interfaces.music5; 

‘import polymorphism.music.Note; 

‘import static net.mindview.util.Print.*; 


interface Instrument { 

// Compile-time constant: 

int VALUE = 5; // static & final 

// Cannot have method definitions: ， 

void play(Note n); // Automatically public 

void adjust(); . 
} 


class Wind implements Instrument { 
public void play(Note n) { 
print(this + ".play() ”+ n)i 





} 

public String toString() { return "Wind"; } 

public void adjust() { print(this + *.adjust()"); } 
} 


class Percussion implements Instrument { 
public void play(Note n) { 
print(this + ".play() " +n); 


} 

public String toString() { return "Percussion"; } 

public void adjust() { print(this + ".adjust()"); } 
} 


class Stringed implements Instrument { 
public void play(Note n) ¢ 
print(this + ".play() " + n); 


} 

public String toString() { return "Stringed"; } 

public void adjust() { print(this + ".adjust()"); } = 
} 


class Brass extends Wind { 
public String toString() { return "Brass"; } 
} 


class Woodwind extends Wind { 
public String toString() { return "Woodwind"; } 
上 
public class Music5 { 
// Doesn't care about type, so new types 
// added to the system still work right: 
static void tune(Instrument i) { 
Wo. 
1. play (Note. MIDDLE_C); 
} 


static void tuneAll(Instrument[] e) { 
for(Instrument i : e) 
tune(i); . 
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public static void main(String[} args) { 
// Upcasting during ‘addition to the array: 
Instrument[{] orchestra = { 
new Wind(), 
new Percussion(), 
new Stringed(). 
new Brass(), 
new Woodwind() 


yi 
tuneAll (orchestra); 

} A Output: 

Wind.play() MIDDLE_C 

Percussion.play() MIDDLE_C 

Stringed.play() MIDDLE_C 

Brass.play() MIDDLE_C 

Woodwind.play() MIDDLE_C 

AAA 

此 实例 的 这 个 版 本 还 有 另外 一 处 改动 : what( 方 法 已 经 被 修改 为 toString() 方 法 ， 因 为 
toString0 的 逻辑 正 是 what0 要 实现 的 逻辑 。 由 于 toString0 方 法 是 根 类 Object 的 一 部 分 ， 因 此 它 
不 需要 出 现在 接口 中 。 

余下 的 代码 其 工作 方式 都 是 相同 的 。 无 论 是 将 其 向 上 转型 为 称 为 Instrument 的 普通 类 ， 还 
是 称 为 Instrument 的 抽象 类 , 或 是 称 为 Instrument 的 接口 ,都 不 会 有 问题 。 它 的 行为 都 是 相同 的 。 
事实 上 ， 你 可 以 在 tune0 方 法 中 看 到 ， 没 有 任何 依据 来 证 明 Instrument 是 一 个 普通 类 、 抽 象 类 ， 
还 是 一 个 接口 。 

练习 5，(2) 在 某 个 包 内 创建 一 个 接口 ， 内 含 三 个 方法 ， 然 后 在 另 一 个 包 中 实现 此 接口 。 

练习 6，(2) 证 明 接口 内 所 有 的 方法 都 自动 是 public 的 。 

练习 7: (1) 修改 第 8 章 中 的 练习 9， 使 Rodent 成 为 一 个 接口 。 

练习 8: (2) 在 polymorphism.Sandwichjava 中 ， 创 建 接口 FastFood 并 添加 合适 的 方法 ， 然 后 
修改 Sandwich 以 实现 FastFood 接 口 。 

练习 9: (3) 重 构 Music5.java， 将 在 Wind、Percussion 和 Stringed 中 的 公共 方法 移入 一 个 抽 
象 类 中 。 

练习 10: (3) 修改 Musics.java， 添 加 Playable 接 口 。 将 play0 的 声明 从 Instrument 中 移 到 
Playable 中 。 通 过 将 Playable 包 括 在 implements 列 表 中 ,把 Playable 添 加 到 导出 类 中 。 修 改 tune()， 
使 它 接受 Playable 而 不 是 Instrument 作 为 参数 。 


9.3 TERE 


只 要 一 个 方法 操作 的 是 类 而 非 接口 ， 那 么 你 就 只 能 使 用 这 个 类 及 其 子 类 。 如 果 你 想 要 将 这 
个 方法 应 用 于 不 在 此 继承 结构 中 的 某 个 类 ， 那 么 你 就 会 触 老头 了 。 接 口 可 以 在 很 大 程度 上 放宽 
这 种 限制 ， ， 它 使 得 我 们 可 以 编写 可 复 用 性 更 好 的 代码 。 

例如 ， 假 设 有 一 个 Processor 类 ， 它 有 一 个 hame0 方 法 ， 另 外 还 有 一 个 process0 方 法 ， 该 方 
法 接受 输入 参数 ， 修 改 它 的 值 ， 然 后 产生 输出 。 这 个 类 做 为 基 类 而 被 扩展 ， 用 来 创建 各 种 不 同 
类 型 的 Processor。 在 本 例 中 ，Processor 的 子 类 将 修改 String 对 象 注意 ， 返 回 类 型 可 以 是 协 变 
类 型 ， 而 非 参 数 类 型 ) : 


//: interfaces/classprocessor/Apply. java 
package interfaces .classprocessor; 
import java.util.*; 

import static net.mindview.util.Print.*; 
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class Processor { 
public String name() { 
return getClass().getSimpleName(); 


, 
Object process(Object input) { return input; } 
} 


class Upcase extends Processor { 
String process(Object input) { // Covariant return 
return ((String) input) . toUpperCase() ; 。 
} 
} 


class Downcase extends Processor { 
String process(Object input) { 
return ((String) input) .toLowerCase() ; 
} 
} 


class Splitter extends Processor { 
String process(Object input) { 
// The split() argument divides a String into pieces: 
return Arrays. toString(((String) input) .split(" ")); 


} 


public class Apply { 
public static void process (Processor p, Object s) { 
print("Using Processor ”+ p.name()): 
print (p.process(s)); 
} 
public static String s = 
"Disagreement with beliefs is by definition incorrect"; 
public static void main(String[] args) { 
Process(new Upcase(), $); 
Process(new Downcase(), $); 
process(new Splitter(), s); 


} 
} /* Output: 
Using Processor Upcase 
DISAGREEMENT WITH BELIEFS IS BY DEFINITION INCORRECT 
Using Processor Downcase 
disagreement with beliefs is by definition incorrect 
Using Processor Splitter 
(Disagreement, with, beliefs, is, by, definition, 
incorrect] 
hn 


Apply.process0 方 法 可 以 接受 任何 类 型 的 Processor， 并 将 其 应 用 到 一 个 Object 对 象 上 ， 然 后 


打印 结果 。 像 本 例 这样 ， 创 建 一 个 能 够 根据 所 传递 的 参数 对 象 的 不 同 而 具有 不 同行 为 的 方法 ， 


被 称 为 策略 设计 模式 。 这 类 方法 包含 所 要 执行 的 算法 中 固定 不 变 的 部 分 ， 而 “策略 ”包含 变化 
的 部 分 。 策 略 就 是 传递 进去 的 参数 对 象 ， 它 包含 要 执行 的 代码 。 这 里 ，Processor 对 象 就 是 一 个 


策略 ， 在 main0 中 可 以 看 到 有 三 种 不 同类 型 的 策略 应 用 到 了 String 类 型 的 s 对 象 上 。 


split0 方 法 是 String 类 的 一 部 分 ， 它 接受 String 类 型 的 对 象 ， 并 以 传递 进来 的 参数 作为 边界 ， 
将 该 String 对 象 分 隔 开 ,然后 返回 一 个 数组 String[]。 它 在 这 里 被 用 来 当 作 创建 String 数 组 的 快捷 


方式 。 


现在 假设 我 们 发 现 了 一 组 电子 滤波 器 ， 它 们 看 起 来 好 像 适 用 于 Apply.process0 方 法 : 


//: interfaces/filters/Waveform.java 
package interfaces.filters; 


public class Waveform { 











321 

















176 BIE 





private static long counter; 

private final long id = counter++; 

public String toString() { return “Waveform "+ id; } 
} M~ 


11: interfaces/filters/Filter.java 
package interfaces. filters; 


vi 


public class Filter { . 
public String name() { 
return getClass().getSimpleName() ; m 
} 
public Waveform process (Waveform input) { return input; } 
} Mi~ 


//: interfaces/filters/LowPass.java 
package interfaces.filters; 


Public class LowPass extends Filter { 
double cutoff; 
public LowPass(double cutoff) { this.cutoff = cutoff; } 
public Waveform process (Waveform input) { 
return input; // Dummy processing 
} 
} Mi~ 


//: interfaces/filters/HighPass. java 
package interfaces. filters; 


public class HighPass extends Filter { 
double cutoff; 
public HighPass(double cutoff) { this.cutoff = cutoff; } 
public Waveform process (Waveform input) { return input; } 
} I~ 


//: interfaces/filters/BandPass. java 
package interfaces.filters; 


public class BandPass extends Filter { 
double lowCutoff, highCutoff; 
public BandPass (double lowCut, double highCut) { 
lowCutoff = lowCut; 
highCutoff = highcut; 
} 
public Waveform process(Waveform fnput) { return iaput; ) 
Uh 


Filter 与 Processor 具 有 相同 的 接口 元 素 ， 但 是 因为 它 并 非 继承 自 Processor 一 一 因为 Filter 类 
的 创建 者 压根 不 清楚 你 想 要 将 它 用 作 Processor 一 一 因此 你 不 能 将 Filter 用 于 Apply.process( 方 法 ， 
即便 这 样 做 可 以 正常 运行 。 这 里 主要 是 因为 Apply:process0 方 法 和 Processor 之 间 的 耦合 过 紧 ， 
已 经 超出 了 所 需要 的 程度 ， 这 就 使 得 应 该 复 用 Apply.process0 的 代码 时 ， 复 用 却 被 禁止 了 。 另 外 
还 需要 注意 的 是 它们 的 输入 和 输出 都 是 Waveform。 

但 是 ， 如 果 Processor 是 一 个 接口 ， 那 么 这 些 限制 就 会 变 得 松动 ， 使 得 你 可 以 复 用 结构 该 接 
口 的 Apply.process0。 下 面 是 Erocessor 和 Apply 的 修改 版 本 : 


//: interfaces/interfaceprocessor/Processor .java 
package interfaces. interfaceprocessor: 


public interface Processor { 
String name(); 
Object process(Object input); 
dh 


LLE interfaces/interfaceprocessor/Apply. java 
package interfaces. interfaceprocessor 
import static net.mindview.util.Print.*; 
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public class Apply { 
public static void process(Processor p, Object s) { 
print("Using Processor " + p.name()); 
print(p.process(s)); 
} 
} M~ 


复 用 代码 的 第 一 种 方式 是 客户 端 程序 员 遵循 该 接口 来 编写 他 们 自己 的 类 ， 就 像 下 面 这 样 ， 


//: interfaces/interfaceprocessor/StringProcessor.java 
package interfaces. interfaceprocessor; , 
import java.util.*; 


public abstract class StringProcessor implements Processor{ 
public String name() { 
return getClass().getSimpleName() ; 
} 
public abstract String process (Object input); 
public static String s = 
"If she weighs the same as a duck, she’s made of wood"; 
public static void main(String[] args) { 
Apply.process(new Upcase(), $); 
Apply.process(new Downcase(), S); 
Apply.process(new Splitter(), 5); > 


) 


class Upcase extends StringProcessor { ú 
public String process(Object input) { // Covariant return 
return ((String) input) . toUpperCase(); 
} 
} 


class Downcase extends StringProcessor { 
public String process(Object input) { 
return ((String) input). toLowerCase(); > 
de 
} 


class Splitter extends StringProcessor { 
public String process (Object input) { 
return Arrays. toString(((String) input).split(" ")); 
} 
} /* Output: 
Using Processor Upcase . 
IF SHE WEIGHS THE SAME AS A DUCK, SHE'S MADE OF WOOD 
Using Processor Downcase 
if she weighs the same as a duck, she's made of wood 
Using Processor Splitter 
[If, she, weighs, the, same, as, a, duck,, she's, made, of, 
wood] 
WMh~ 


但 是 ， 你 经 常 磁 到 的 情况 是 你 无 法 修改 你 想 要 使 用 的 类 。 例 如 ， 在 电子 滤波 器 的 例子 中 ， 
类 库 是 被 发 现 而 非 被 创建 的 。 在 这 些 情况 下 ， 可 以 使 用 适配器 设计 模式 。 适 配器 中 的 代码 将 接 


受 你 所 拥有 的 接口 ， 并 产生 你 所 需要 的 接口 ， 就 像 下 面 这 样 : 


//: interfaces/interfaceprocessor/FilterProcessor. java 
package interfaces. interfaceprocessor; 
import interfaces. filters.*; 


class FilterAdapter implements Processor { 
Filter filter; 
public FilterAdapter(Filter filter) { 
this. filter = filter; 
¥ 
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public String name() { return filter.name(); } 
public Waveform process(Object input) { 
return filter.process( (Waveform) input) ; 


} 


public class FilterProcessor { 
public static void main(String[] args) { 
Waveform w = new Waveform(); 
Apply. process (new FilterAdapter (new LowPass(1.)), w); 
Apply.process(new FilterAdapter (new HighPass(2.8)), W); 
Apply. process( 
new FilterAdapter(new BandPass(3.0, 4.0)). w); 


} 

} /* Output: 

Using Processor LowPass 

Waveform @ 

Using Processor HighPass 

Waveform @ 

Using Processor BandPass 
` Waveform 9 

Wha 


在 这 种 使 用 适配器 的 方式 中 ，FilterAdapter 的 构造 器 接受 你 所 拥有 的 接口 Filter， 然 后 生成 
具有 你 所 需要 的 Processor 接 口 的 对 象 。 你 可 能 还 注意 到 了 ， 在 FilterAdapter 类 中 用 到 了 代理 。 

将 接口 从 具体 实现 中 解 耦 使 得 接口 可 以 应 用 于 多 种 不 同 的 具体 实现 ， 因 此 代码 也 就 更 具 可 
复 用 性 。 

练习 11: (4) 创建 一 个 类 ， 它 有 一 个 方法 用 于 接受 一 个 String 类 型 的 参数 ， 生 成 的 结果 是 将 
该 参数 中 每 一 对 字符 进行 互 换 。 对 该 类 进行 适 配 ， 使 得 它 可 以 用 于 interfaceprocessor.Apply. 
process), 


9.4 Java 中 的 多 重 继承 


接口 不 仅仅 只 是 一 种 更 纯粹 形式 的 抽象 类 ， 它 的 目标 比 这 要 高 。 因 为 接口 是 根本 没有 任何 
具体 实现 的 一 一 也 就 是 说 ， 没 有 任何 与 接口 相关 的 存储 ， 因此， 也 就 无 法 阻止 多 个 接口 的 组 合 。 
这 一 点 是 很 有 价值 的 ， 因 为 你 有 时 需要 去 表示 “一 个 x 是 一 个 a 和 一 个 b 以 及 一 个 c" 。 在 C++ 中 ， 
组 合 多 个 类 的 接口 的 行为 被 称 作 多 重 继承 。 它 可 能 会 使 你 背负 很 沉重 的 包 宰 ， 因 为 每 个 类 都 有 
一 个 具体 实现 。 在 Java 中 ， 你 可 以 执行 相同 的 行为 ， 但 是 只 有 一 个 类 可 以 有 具体 实现 ， 因 此， 
通过 组 合 多 个 接口 ，C++ 中 的 问题 是 不 会 在 Java 中 发 生 的 : 








|L ar} 
抽象 基 类 或 具体 基 类 pr aP 


= S | am j., 
| 接 Hn | 
| | | i 
{ 基 类 方法 | ao | 接 m2 [| nn | 
在 导出 类 中 ， 不 强制 要 求 必须 有 一 个 是 抽象 的 或 “具体 的 ”( 没 有 任何 抽象 方法 的 ) 基 类 。 
如 果 要 从 一 个 非 接口 的 类 继承 ， 那 么 只 能 从 一 个 类 去 继承 。 其 佘 的 基 元 素 都 必须 是 接口 。 需 要 
将 所 有 的 接口 名 都 置 于 implements 关 键 字 之 后 ， 用 逗号 将 它们 一 一 隔 开 。 可 以 继承 任意 多 个 接 


口 ， 并 可 以 向 上 转型 为 每 个 接口 ， 因 为 每 一 个 接口 都 是 一 个 独立 类 型 。 下 面 的 例子 展示 了 一 个 
具体 类 组 合 数 个 接口 之 后 产生 了 一 个 新 类 : 
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//: interfaces/Adventure. java 
// Multipie interfaces. 


interface CanFight { 
void fight(); 
} 


interface CanSwim { 
void swim(); 
未 


interface CanFly { 
void fly: 
} 


class ActionCharacter { 
public void fight() {} 
} 


class Hero extends ActionCharacter 
implements CanFight, CanSwim, CanFly { 
public void swim() {} 
public void fly() {} 
} 


public class Adventure { 

public static void t(CanFight x) { x.fight(); } 
public static void u(CanSwim x) { x.swim(); } 
public static void v(CanFly x) { x.fly(); } 
public static void w(ActionCharacter x) { x.fight(); } 
public static void main(String{] args) { 

Hero h = new Hero(); 

t(h); // Treat it as a CanFight 

u(h); // Treat it as a CanSwim 

v(h); // Treat it as a CanFly 

wih); // Treat it as an ActionCharacter 

} Mite 

可 以 看 到 ，Hero 组 合 了 具体 类 ActionCharacter 和 接口 CanFight、CanSwim 和 CanFly。 当 通 
过 这 种 方式 将 一 个 具体 类 和 多 个 接口 组 合 到 一 起 时 ， 这 个 具体 类 必须 放 在 前 面 ， 后 面 跟着 的 才 
是 接口 (否则 编译 器 会 报错 )。 

注意 ，CanFight 接 口 与 ActionCharacter 类 中 的 fight0 方 法 的 特征 签名 是 一 样 的 ， 而且 ,在 
Hero 中 并 没有 提供 fight0 的 定义 。 可 以 扩展 接口 ， 但 是 得 到 的 只 是 另 一 个 接口 。 当 想 要 创建 对 
象 时 ， 所 有 的 定义 首先 必须 都 存在 。 即 使 Hero 没 有 显 式 地 提供 fightO 的 定义 ， 其 定义 也 因 
ActionCharacter 而 随 之 而 来 ， 这 样 就 使 得 创建 Hero 对 象 成 为 了 可 能 。 

在 Adventure 类 中 ， 可 以 看 到 有 四 个 方法 把 上 述 各 种 接口 和 具体 类 作为 参数 。 当 Hero 对 象 被 
创建 时 ， 它 可 以 被 传递 给 这 些 方法 中 的 任何 一 个 ， 这 意味 着 它 依次 被 向 上 转型 为 每 一 个 接口 。 
由 于 Java 中 这 种 设计 接口 的 方式 ， 使 得 这 项 工作 并 不 需要 程序 员 付出 任何 特别 的 努力 。 

一 定 要 记 住 ， 前 面 的 例子 所 展示 的 就 是 使 用 接口 的 核心 原因 : 为 了 能 够 向 上 转型 为 多 个 基 
类 型 〈 以 及 由 此 而 带 来 的 灵活 性 ) 。 然 而 ， 使 用 接口 的 第 二 个 原因 却 是 与 使 用 抽象 基 类 相同 ; 防 
止 客户 端 程序 员 创建 该 类 的 对 象 ， 并 确保 这 仅仅 是 建立 一 个 接口 。 这 就 带 来 了 一 个 问题 : 我 们 
应 该 使 用 接口 还 是 抽象 类 ? 如果 要 创建 不 带 任何 方法 定义 和 成 员 变 量 的 基 类 ， 那 么 就 应 该 选择 
接口 而 不 是 抽象 类 。 事 实 上 ， 如 果 知道 某 事 物 应 该 成 为 一 个 基 类 ， 那 么 第 一 选择 应 该 是 使 它 成 
为 一 个 接口 〈 该 主题 在 本 章 的 总 结 中 将 再 次 讨论 ) 。 

练习 12: (2) 在 Adventurejava 中 ， 按 照 其 他 接口 的 样式 ， 增 加 一 个 CanClimb 接 口 。 
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练习 13:， (2) 创建 一 个 接口 ， 并 从 该 接口 继承 两 个 接口 ， 然 后 从 后 面 两 个 接口 多 重 继承 第 三 
kOe. 


9.5 通过 继承 来 扩展 接口 


通过 继承 ， 可 以 很 容易 地 在 接口 中 添加 新 的 方法 声明 ， 还 可 以 通过 继承 在 新 接口 中 组 合 数 
个 接口 。 这 两 种 情况 都 可 以 获得 新 的 接口 ， 就 像 在 下 例 中 所 看 到 的 ; 


//: interfaces/HorrorShow. java 
/1 Extending an interface with inheritance. 


interface Monster { 
void menace(); 
} 


interface DangerousMonster extends Monster { 
void destroy(); 
} 


interface Lethal { 
void KINO; 
} 


class DragonZilla implements DangerousMonster { 
public void menace() {} 
public void destroy() {} 

} 


interface Vampire extends DangerousMonster, Lethal { 
void drinkBlood(); 
} 


class VeryBadVampire implements Vampire { 
public void menace() {} 
public void destroy() {} 
public void kill() {} 
public void drinkBlood() {} 
} 


public class HorrorShow { 
static void u(Monster b) { b.menace(); } 
static void v(DangerousMonster d) { 
d.menace(); 


d.destroy(); 


static void w(Lethal 1) { 1.kill(); } 

Public static void main(String[] args) { 
Dangeroustonster barney = new DragonZitla(); 
u(barney) ; 

v (barney) ; 

Vampire vlad = new VeryBadVampire(): 
u(vlad) ; 

v(vlad) ; 

w(vlad): 


} 
¥ Mi~ 


DangerousMonster 是 Monster 的 直接 扩展 ， 它 产生 了 一 个 新 接口 。DragonZilla 中 实现 了 这 


个 接口 。 


在 Vampire 中 使 用 的 语法 仅 适用 于 接口 继承 。 一 般 情况 下 ， 只 可 以 将 extends 用 于 单一 类 ， 


日 ”这 可 以 说 明 接口 是 如 何 防止 在 C++ 多 重 继 承 中 所 产生 的 “ 萎 形 问题 ”的 。 
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但 是 可 以 引用 多 个 基 类 接口 。 就 像 所 看 到 的 ， 只 需 用 逗号 将 接口 名 一 一 分 隔 开 即 可 。 

练习 14: (2) 创建 三 个 接口 ， 每 个 接口 都 包含 两 个 方法 。 继 承 出 一 个 接口 ， 它 组 合 了 这 三 个 
接口 并 添加 了 一 个 新 方法 。 创 建 一 个 实现 了 该 新 接口 并 且 继承 了 某 个 具体 类 的 类 。 现 在 编写 四 
个 方法 ， 每 一 个 都 接受 这 四 个 接口 之 一 作为 参数 。 在 main0 方 法 中 ， 创 建 这 个 类 的 对 象 ， 并 将 
其 传递 给 这 四 个 方法 。 

练习 15: (2) 将 前 一 个 练习 修改 为 : 创建 一 个 抽象 类 ， 并 将 其 继承 到 一 个 导出 类 中 。 


9.5.1 组 合 接口 时 的 名 字 冲 突 


在 实现 多 重 继承 时 ， 可 能 会 碰 到 一 个 小 陷阱 。 在 前 面 的 例子 中 ，CanFight 和 ActionCharacter 
都 有 一 个 相同 的 void fight0 方 法 。 这 不 是 问题 所 在 ， 因 为 该 方法 在 二 者 中 是 相同 的 。 相 同 的 方法 
不 会 有 什么 问题 ， 但 是 如 果 它 们 的 签名 或 返回 类 型 不 同 ， 又 会 怎么 样 呢 ? 这 有 一 个 例子 ; 

//: interfaces/InterfaceCollision. java 

package interfaces; 


interface I1 { void fO; } 

` interface 12 { int f(int i): } 
interface 13 { int fO; } 
class C { public int f() { return 1; } } 


class C2 implements 11, 12 { 
public void fO {} 
public int f(int i) { return 1; } // overloaded 
+ 


class C3 extends C implements I2 ( 
public int f(int i) { return 1; } // overloaded 
} 


class C4 extends C implements 13 { 
// Identical, no problem: 
public int f() { return 1; } 

$ 


// Methods differ only by return type: 
//! class C5 extends C implements I1 {} 
//! interface I4 extends I1, 13 {} ///:~ ` 


此 时 困难 来 了 ， 因 为 覆盖 、 实 现 和 重 载 令 人 不 快 地 搅 在 了 一 起 ， 而 且 重 载 方法 仅 通过 返回 
类 型 是 区 分 不 开 的 。 当 撤销 最 后 两 行 的 注释 时 ， 下 列 错误 消息 就 说 明了 这 一 切 : 

InterfaceCollision.java:23: f() in C cannot implement f() in I1; attempting 

to use incompatible return type 

found: int 

required: void 

InterfaceCollision,java:24: Interfaces I3 and I1 are incompatible; both 

define ft ), but with different return type 


在 打算 组 合 的 不 同 接口 中 使 用 相同 的 方法 名 通常 会 造成 代码 可 读 性 的 混乱 ， 请 尽量 避免 这 
种 情况 。 


9.6 适 配 接口 


接口 最 吸引 人 的 原因 之 一 就 是 允许 同一 个 接口 具有 多 个 不 同 的 具体 实现 。 在 简单 的 情况 中 ， 
它 的 体现 形式 通常 是 一 个 接受 接口 类 型 的 方法 ， 而 该 接口 的 实现 和 向 该 方法 传递 的 对 象 则 取决 
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于 方法 的 使 用 者 。 

因此 ， 接 口 的 一 种 常见 用 法 就 是 前 面 提 到 的 策略 设计 模式 ， 此 时 你 编写 一 个 执行 某 些 操作 
的 方法 ， 而 该 方法 将 接受 一 个 同样 是 你 指定 的 接口 。 你 主要 就 是 要 声明 :“ 你 可 以 用 任何 你 想 要 
的 对 象 来 调用 我 的 方法 ， 只 要 你 的 对 象 遵循 我 的 接口 .” 这 使 得 你 的 方法 更 加 灵活 、 通 用 ， 并 更 
具 可 复 用 性 。 

例如 ，Java SE5 的 Scanner 类 (在 第 13 章 中 就 更 多 地 了 解 它 ) 的 构造 器 接受 的 就 是 一 个 
Readable 接 口 。 你 会 发 现 Readable 没 有 用 作 Java 标 准 类 库 中 其 他 任何 方法 的 参数 ， 它 是 单独 为 
Scanner 创 建 的 ， 以 使 得 Scanmer 不 必 将 其 参数 限制 为 某 个 特定 类 。 通 过 这 种 方式 ，Scanner 可 以 
作用 于 更 多 的 类 型 。 如 果 你 创建 了 一 个 新 的 类 ， 并 且 想 让 Scanner 可 以 作用 于 它 ， 那 么 你 就 应 该 
让 它 成 为 Readable， 就 像 下 面 这 样 : 


//: interfaces/RandomWords.java 

// Implementing nterface to conform to a method. 
import java.nio 
import java.uti 





public class RandomWords implements Readable { 
private static Random rand = new Random(47) ; 
private static final char[] capitals = 
“ABCDEFGHI JKLMNOPQRSTUVWXYZ" . toCharArray(); 
private static final char[] lowers = 
"abcde fghijklmnopqrstuvwxyz”. toCharArray(); 
private static final char{] vowels = 
"aeiou". toCharArray(); 
private int count; 
public RandomWords (int count) { this.count = count; } 
public int read(CharBuffer cb) { 
if(count-- == @) 
return -1; // Indicates end of input 
cb. append(capitals[rand.nextInt (capitals. length) ]); 
for(int 1 = 0; 1 < 4; i++) { 
cb. append(vowels{rand.nextInt (vowels. length)]); 
cb. append(lowers (rand. nextInt (lowers. length)]); 


} 
cb.append(" "); 
return 10; // Number of characters appended 


} 
public static void main(String{] args) { 
Scanner s = new Scanner (new RandomWords(10)); 
while(s.hasNext()) 
System.out.printin(s.next()); 


} 

} /* Output: 
Yazeruyac 
Fowenucor 
Goeazimom 
Raewuacio 
Nuoades iw 
Hageaikux 
Ruqicibui 
Numasetih 
Kuuuuozog 
Waqizeyoy 
Wh 


Readable 接 口 只 要 求实 现 read0 方 法 ， 在 read0 内 部 ， 将 输入 内 容 添加 到 CharBuffer 参 数 中 
(有 多 种 方法 可 以 实现 此 目的 ， 请 查看 CharBuffer 的 文档 ) ， 或 者 在 没有 任何 输入 时 返回 一 1。 

假设 你 有 一 个 还 未 实现 Readable 的 类 ， 怎 样 才 能 让 Scanner 作 用 于 它 呢 ? 下 面 这 个 类 就 是 一 
个 例子 ， 它 可 以 产生 随机 浮 点 数 : 
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J/: interfaces/RandomDoubles. java 
import java.util.*; 


public class RandomDoubles { 
private static Random rand = new Random(47) ; 
public double next() { return rand.nextDouble(); } 
public static void main(String(] args) { 
RandomDoubles rd = new RandomDoubles(); 
for(int 1 = @; i < 7; 1 ++) 
System.out.print(rd.next() + " "): 


} 
} /* Output: 
@.7271157860730044 9.5369454568634242 ©. 16620656493302599 
9.18847866977771732 @.5166026801268457 ©.2678662084200585 
0.2613610344283964 
Whim 


我 们 再 次 使 用 了 适配器 模式 ， 但 是 在 本 例 中 ， 被 适 配 的 类 可 以 通过 继承 和 实现 Readable 接 
口 来 创建 。 因 此 ， 通 过 使 用 interface 关 键 字 提 供 的 伪 多 重 继承 机 制 ， 我 们 可 以 生成 既是 Random- 
Doubles 又 是 Readable 的 新 类 : 


//: interfaces/AdaptedRandomDoubles . java 
// Creating an adapter with inheritance. 
import java.nio 
import java.util 





public class AdaptedRandomDoubles extends RandomDoubles 
implements Readable { 
private int count; 
public AdaptedRandomDoubles(int count) { 
this.count = count; 


} 
public int read(CharBuffer cb) { 
if(count-- == @) 
return -1; 
String result = Double.toString(next()) + * "; 
cb. append(result); 
return result.length(); 
} 
public static void main(String{] args) { 
Scanner s = new Scanner (new AdaptedRandomDoubles(7)): 
whi le(s.hasNextDouble()) 
System.out.print(s.nextDouble() + * "); 


} /* Output: 

@.7271157860730044 @.5309454508634242 @.16620656493302599 
0. 18847866977771732 @.5166620801268457 @.2678662684200585 
©. 2613610344283964 

“i~ 


因为 在 这 种 方式 中 ， 我 们 可 以 在 任何 现 有 类 之 上 添加 新 的 接口 ， 所 以 这 意味 着 让 方法 接受 
接口 类 型 ， 是 一 种 让 任何 类 都 可 以 对 该 方法 进行 适 配 的 方式 。 这 就 是 使 用 接口 而 不 是 类 的 强大 
之 处 。 

练习 16: (3) 创建 一 个 类 ， 它 将 生成 一 个 char 序 列 ， 适 配 这 个 类 ， 使 其 可 以 成 为 Scanner 对 
象 的 一 种 输入 。 


9.7 接口 中 的 域 


因为 你 放 和 接口 中 的 任何 域 都 自动 是 static 和 final 的 ， 所 以 接口 就 成 为 了 一 种 很 便捷 的 用 来 
创建 常量 组 的 工具 。 在 Java SE5 之 前 ， 这 是 产生 与 C 或 C++ 中 的 enum (HRR) 具有 相同 效果 
的 类 型 的 唯一 途径 。 因 此 在 Java SE5 之 前 的 代码 中 你 会 看 到 下 面 这 样 的 代码 ; 
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//: interfaces/Months. java 
71 Using interfaces to create groups of constants. 
package interfaces; 


public interface Months { 
int 
JANUARY = 1, FEBRUARY = 2, MARCH = 3, 
APRIL = 4, MAY = 5, JUNE = 6, JULY = 7, 
AUGUST = 8, SEPTEMBER = 9, OCTOBER = 10, 
NOVEMBER = 11, DECEMBER = 12; 
} Mi~ 


请 注意 ，Java 中 标识 具有 常量 初始 化 值 的 static final 时 ， 会 使 用 大 写字 母 的 风格 (在 一 个 标 
识 符 中 用 下 划 线 来 分 隔 多 个 单词 )。 接 口中 的 域 自动 是 public 的 ， 所 以 没有 显 式 地 指明 这 一 点 。 

有 了 Java SE5， 你 就 可 以 使 用 更 加 强大 而 灵活 的 enum 关 键 字 ， 因 此 ， 使 用 接口 来 群 组 常 
量 已 经 显得 没什么 意义 了 。 但 是 ， 当 你 阅读 遗留 的 代码 时 ， 在 许多 情况 下 你 可 能 还 是 会 磁 到 
这 种 旧 的 习惯 用 法 (www.MindView.net 上 关于 本 书 的 补充 材料 中 ， 包 含有 关 在 Java SE5 之 前 
使 用 接口 来 生成 枚 举 类 型 的 方式 的 完整 描述 )。 在 第 19 章 中 可 以 看 到 更 多 的 关于 使 用 enum 的 细 
节 说 明 。 

练习 17: (2) 证 明 在 接口 中 的 域 隐 式 地 是 static 和 final 的 。 


9.7.1 初始 化 接口 中 的 域 
在 接口 中 定义 的 域 不 能 是 “ 空 final”"， 但 是 可 以 被 非常 量 表达 式 初始 化 。 例 如 : 


//: interfaces/RandVats. java 
// Initializing interface fields with 
// non-constant initializers. 

import java.util.*; 


public interface RandVals { 
Random RAND = new Random(47); 
int RANDOM_INT = RAND .nextInt(19) 
long RANDOM_LONG = RAND.nextLong() * 10; 
float RANDOM_FLOAT = RAND.nextLong() * 10; 
double RANDOM_OOUBLE = RAND.nextOouble() * 10; 
} Mi~ 


既然 域 是 static 的 ， 它 们 就 可 以 在 类 第 一 次 被 加 载 时 初始 化 ， 这 发 生 在 任何 域 首次 被 访问 时 。 
这 里 给 出 了 一 个 简单 的 测试 : 


//: interfaces/TestRandVals. java 
import static net.mindview.util.Print.*; 


public class TestRandvals { 
public static void main(String{] args) { 
print (RandVals .RANDOM_INT) ; 
Print (RandVals. RANDOM_LONG) + 
print (RandVals. RANDOM FLOAT) ; 
print (RandVals.RANDOM_DOUBLE) ; 


} 
} /* Output: 
8 


-32032247016559954 
-8.5939291E18 
5.779976127815049 
Whim 


当然 ， 这 些 域 不 是 接口 的 一 部 分 ， 它 们 的 值 被 存储 在 该 接口 的 静态 存储 区 域内 。 








# oa 185 





98 WERO 
SECURE ERITH H. KRR TH SAR ARRE: 


//: interfaces/nesting/NestingInterfaces. java 
package interfaces nesting: 336] 
class A { 
interface B { 
”void f0; 
} 
public class BImp implements B { 
public void f() {} 
$ 
private class BImp2 implements B { 
public void f() {} 














} 

public interface C { 
void f(); 

} 

Class CImp implements C { 
public void fO {} 

} 

private class CImp2 implements C { 
public void fO {} 

ig 

private interface D { 
void t0; 

$ 

private class DImp implements D { 
public void +() {} 


} 

public class DImp2 implements D { 
public void fO {} 

} 

public D getD() { return new DImp2(); } 

private D dRef; 

public void receiveD(D d) { 
dRef = d; 
dRet.t(): 

} 

} 


interface E { 

interface G { 
void fO: 

} 

/7 Rédundant "public": 

public interface H { 
void f(); . 337] 

t 

void g(); 

// Cannot be private within an interface: 

//! private interface I {} 














2 


public class NestingInterfaces { 
public class BImp implements A.B { 
public void f() {} 
} 
class CImp implements A.C { 
public void f() (} 
} 
// Cannot implement a private interface except 


日 感谢 Martin Danner 在 研讨 会 上 提出 这 个 问题 。 





338) 
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// within that interface's defining class: 
/AL class DImp implements A.D { 
44! public void FO {} 
my 
class EInp implements E { 
public void g0 {} 


} 
class EGImp implements E.G { 
public void fO {} 


} 
class EImp2 implements E { 
public void g() {} 
class EG implements E.G { 
public void f() {} 
} 
J: 
public static void main(String{] args) { 
A a = new AQ); 
// Can't access A.D: 
//! A.D ad = a.getD(); 
// Doesn't return anything but A.D: 
/AL A.DImp2 di2 = a.getD(); 
// Cannot access a member of the interface: 
1/! a.getD() .fF(); 
// Only another A can do anything with getD(): 
A a2 = new AQ: 
a2. receiveD(a.getD()); 


) 

} /LA 

在 类 中 伐 套 接口 的 语法 是 相当 显而易见 的 ， 就 像 非 铭 套 接 口 一 样 ， 可 以 拥有 public 和 “ 包 访 
间 ” 两 种 可 视 性 。 

作为 一 种 新 添加 的 方式 ， 接 口 也 可 以 被 实现 为 private 的 ， 就 像 在 A.D 中 所 看 到 的 (相同 的 语 
法 既 适 用 于 侈 套 接口 ， 也 适用 于 供 套 类 )。 那 么 private 的 供 套 接口 能 带 来 什么 好 处 呢 ? 读者 可 能 
会 猜想 ， 它 只 能 够 被 实现 为 DImp 中 的 一 个 private 内 部 类 ， 但 是 A.DImp2 展 示 了 它 同样 可 以 被 实 
现 为 public 类 。 但 是 ，A.DImp2 只 能 被 其 自身 所 使 用 。 你 无 法 说 它 实 现 了 一 个 Private 接口 D。 
此 ， 实 现 一 个 private 接 口 只 是 一 种 方式 ， 它 可 以 强制 该 接口 中 的 方法 定义 不 要 添加 任何 类 型 信 
息 (也 就 是 说 ， 不 允许 向 上 转型 )。 

getD0 方 法 使 我 们 陷入 了 一 个 进退 两 难 的 境地 ， 这 个 问题 与 private 接 口 相 关 ， 它 是 一 个 返回 
对 private 接 口 的 引用 的 public 方 法 。 你 对 这 个 方法 的 返回 值 能 做 些 什么 呢 ? 在 main0 中 ， 可 以 看 
到 数 次 尝试 使 用 返回 值 的 行为 都 失败 了 。 只 有 一 种 方式 可 成 功 ， 那 就 是 将 返回 值 交 给 有 权 使 用 
它 的 对 象 。 在 本 例 中 ， 是 另 一 个 A 通过 receiveD0 方 法 来 实现 的 。 

接口 E 说 明 接口 彼此 之 间 也 可 以 延 套 。 然 而 ， 作 用 于 接口 的 各 种 规则 ， 特 别 是 所 有 的 接口 元 
素 都 必须 是 public 的 ， 在 此 都 会 被 严格 执行 。 因 此 ， ee 自动 就 是 public 
的 ， 而 不 能 声明 为 private 的 。 

NestingInterfaces 展 示 了 供 套 接口 的 各 种 实现 方式 。 特别 要 注意 的 是 ， 当 实 现 某 个 接口 时 ， 
并 不 需要 实现 伐 套 在 其 内 部 的 任何 接口 。 而 且 ，private 接 口 不 能 在 定义 它 的 类 之 外 被 实现 。 

添加 这 些 特性 的 最 初 原因 可 能 是 出 于 对 严格 的 语法 一 致 性 的 考虑 ， 但 是 我 总 认为 ， 一 旦 你 
了 解 了 某 种 特性 ， 就 总 能 够 找到 它 的 用 武之 地 。 


9.9 接口 与 工厂 
接口 是 实现 多 重 继承 的 途径 ， 而 生成 遵循 某 个 接口 的 对 象 的 典型 方式 就 是 工厂 方法 设计 模 
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式 。 这 与 直接 调用 构造 器 不 同 ， 我 们 在 工厂 对 象 上 调用 的 是 创建 方法 ， 而 该 工厂 对 象 将 生成 接 
口 的 某 个 实现 的 对 象 。 理 论 上 ， 通 过 这 种 方式 ， 我 们 的 代码 将 完全 与 接口 的 实现 分 离 ， 这 就 使 
得 我 们 可 以 透明 地 将 某 个 实现 替换 为 另 一 个 实现 。 下 面 的 实例 展示 了 工厂 方法 的 结构 : 


//: interfaces/Factories.java 
import static net.mindview.util.Print.*; 


interface Service { 
void method1() ; 
void method2() ; 

} 


interface ServiceFactory { 
Service getService(); 


} 


class Implementation] implements Service { 
Implementation1() {} // Package access 
Public void method1() {print ("Implementation method1");} 
public void method2() (print(*Implementation1 method2");} 
} 





class ImplementationlFactory implements ServiceFactory { 
public Service getService() { 
return new Implementation1(); 


} 


class Implementation2 implements Service { 
Implementation2() {} // Package access 
public void method1() {print ("Imptementation2 method1") ;} 
public void method2() {print("Implementation2 method2") ;} 
} 


class Implementation2Factory implements ServiceFactory { 
public Service getService() { 
return new Implementation2(); 
} 
} 


public class Factories { 
public static void serviceConsumer(ServiceFactory fact) { 
Service s = fact.getService(): 
$.method2 () ; 
s.method2() ; 


} 

public static void main(String] args) { 
serviceConsumer(new ImplementationiFactory()); 
// Implementations are completely interchangeable: 
serviceConsumer (new Implementation2Factory()) ; 


} 
} /* Output: 
Implementation1 method1 
Implementation] method2 
Implementation? method1 
Implementation? method2 
Wh 


如 果 不 是 用 工厂 方法 ， 你 的 代码 就 必须 在 某 处 指定 将 要 创建 的 Service 的 确切 类 型 ， 以 便 调 
用 合适 的 构造 器 。 

为 什么 我 们 想 要 添加 这 种 额外 级 别 的 间接 性 呢 ? 一 个 常见 的 原因 是 想 要 创建 框架 ;假设 你 
正在 创建 一 个 对 弈 游戏 系统 ， 例 如 ， 在 相同 的 棋盘 上 下 国际 象棋 和 西洋 跳棋 : 


//: interfaces/Games. java 
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// A Game framework using Factory Meth 
import static net.mindview.util.Print. 





interface Game { boolean move(); } 
interface GameFactory { Game getGame(); } 


class Checkers implements Game { a 
private int moves = 0; 
private static final int MOVES 
public boolean move() { 
print("Checkers move " + moves); 
return ++moves != MOVES; 
} 
} 





class CheckersFactory implements GameFactory { 
public Game getGame() { return new Checkers(); } 
} 


class Chess implements Game { 
private int moves = 6; 
private static final int MOVES = 4: 
public boolean move() { 
print("Chess move " + moves); 
return ++moves != MOVES; 
) à 
, 





class ChessFactory implements GameFactory { 
public Game getGame() { return new Chess(); } 
} 


public class Games { 

public static void playGame(GameFactory factory) { 
Game s = factory.getGame(); 
while(s.move()) 

} 

Public static void main(Stringl) args) { 
playGame(new CheckersFactory()); 
playGame(new ChessFactory()); 


} 

} /* Output: 
Checkers move @ 
Checkers move 1 
Checkers move 2 
Chess move @ 
Chess move 1 
Chess move 2 
Chess move 3 
Wha 


如 果 Games 类 表示 一 段 复杂 的 代码 ， 那 么 这 种 方式 就 允许 你 在 不 同类 型 的 游戏 中 复 用 这 段 
代码 。 你 可 以 再 想象 一 些 能 够 从 这 个 模式 中 受益 的 更 加 精巧 的 游戏 。 
在 下 一 章 中 ， 你 将 可 以 看 到 另 一 种 更 加 优雅 的 工厂 实现 方式 ， 那 就 是 使 用 匿名 内 部 类 。 
练习 18: (2) 创建 一 个 Cycle 接口 及 其 Unicycle、Bicycle 和 Tricycle 实 现 。 对 每 种 类 型 的 Cycle 
都 创建 相应 的 工厂 ， 然 后 编写 代码 使 用 这 些 工厂 。 
342] 练习 19:， (3) 使 用 工厂 方法 来 创建 一 个 框架 ， 它 可 以 执行 抛 硬币 和 掷 般 子 功能 。 


9.10 总 结 
“确定 接口 是 理想 选择 ， 因而 应 该 总 是 选择 接口 而 不 是 具体 的 类 这 其 实 是 一 种 引诱 。 当 
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然 ， 对 于 创建 类 ， 几 乎 在 任何 时 刻 ， 都 可 以 替代 为 创建 一 个 接口 和 一 个 工厂 。 

许多 人 都 掉 进 了 这 种 诱惑 的 陷阱 ， 只 要 有 可 能 就 去 创建 接口 和 工厂 。 这 种 逻辑 看 起 来 好 像 
是 因为 需要 使 用 不 同 的 具体 实现 ， 因 此 总 是 应 该 添加 这 种 抽象 性 。 这 实际 上 已 经 变 成 了 一 种 草 
率 的 设计 优化 。 

任何 抽象 性 都 应 该 是 应 真正 的 需求 而 产生 的 。 当 必需 时 ， 你 应 该 重 构 接口 而 不 是 到 处 添加 
额外 级 别 的 间接 性 ， 并 由 此 带 来 的 额外 的 复杂 性 。 这 种 额外 的 复杂 性 非常 显著 ， 如 果 你 让 某 人 
去 处 理 这 种 复杂 性 ， 只 是 因为 你 意识 到 由 于 以 防 万 一 而 添加 了 新 接口 ， 而 没有 其 他 更 有 说 服 力 
的 原因 ， 那 么 好 吧 ， 如 果 我 碰 上 了 这 种 事 ， 那 么 就 会 质疑 此 人 所 作 的 所 有 设计 了 。 

恰当 的 原则 应 该 是 优先 选择 类 而 不 是 接口 。 从 类 开始 ， 如 果 接 口 的 必需 性 变 得 非常 明确 ， 
那么 就 进行 重 构 。 接 口 是 一 种 重要 的 工具 ， 但 是 它们 容易 被 滥用 。 

所 选 习 题 的 答案 都 可 以 在 名 为 The Thinking in Java Annotated Solution Guide 的 电子 文档 中 找 
到 ， 读 者 可 以 从 wwwMindview net 购 买 此 文档 。 a 





第 10 章 内 部 类 


可 以 将 一 个 类 的 定义 放 在 另 一 个 类 的 定义 内 部 ， 这 就 是 内 部 类 。 

内 部 类 是 一 种 非常 有 用 的 特性 ， 因 为 它 允 许 你 把 一 些 逻 辑 相 关 的 类 组 织 在 一 起 ， 并 控制 位 
于 内 部 的 类 的 可 视 性 。 然 而 必须 要 了 解 ， 内 部 类 与 组 合 是 完全 不 同 的 概念 ， 这 一 点 很 重要 。 

在 最 初 ， 内 部 类 看 起 来 就 像 是 一 种 代码 隐藏 机 制 ， 将 类 置 于 其 他 类 的 内 部 。 但 是 ， 你 将 会 
了 解 到 ， 内 部 类 远 不 止 如 此 ， 它 了 解 外 围 类 ， 并 能 与 之 通信 ， 而 且 你 用 内 部 类 写 出 的 代码 更 加 
优雅 而 清晰 ， 尽 管 并 不 总 是 这 样 。 

最 初 ， 内 部 类 可 能 看 起 来 有 些 奇怪 ， 而 且 要 花 些 时 间 才 能 在 设计 中 轻松 地 使 用 它们 。 对 内 
部 类 的 需求 并 非 总 是 很 明显 的 ， 但 是 在 描述 完 内 部 类 的 基本 语法 与 语义 之 后 ，10.8 节 就 应 该 使 得 
内 部 类 的 益处 明确 显现 了 。 

在 10.8 节 之 后 ， 本 章 剩余 部 分 包含 了 对 内 部 类 语法 更 加 详尽 的 探索 ， 这 些 特性 是 为 了 语言 的 
完备 性 而 设计 的 ， 但 是 你 也 许 不 需要 使 用 它们 ， 至 少 一 开始 不 需要 。 因 此 ， 本 章 最 初 的 部 分 也 
许 就 是 你 现在 所 需 的 全 部 ， 你 可 以 将 更 详尽 的 探索 当 作 参 考 资料 。 


10.1 创建 内 部 类 
创建 内 部 类 的 方式 就 如 同 你 想 的 一 样 一 -把 类 的 定义 置 于 外 围 类 的 里 面 : 


//: innerclasses/Parcel1. java 
// Creating inner classes. 


public class Parcell { 
class Contents { 
private int 1 = 11; 
public int value() { return i; } 


class Destination { 
private String label; 
Destination(String whereTo) { 
label = whereTo; 


} 
String: readLabel() { return label; } 
} 
// Using inner classes looks just like 
// using any other class, within Parcell: 
public void ship(String dest) { 
Contents c = new Contents(); 
Destination d = new Destination(dest); 
System.out.printIn(d.readLabel()) ; 


) 

public static void main(String[] args) { 
Parcell p = new Parcell(); 
p.ship("Tasmania"); 


} /* Output: 

Tasmania 

“Ii~ 

当 我 们 在 ship0 方 法 里 面 使 用 内 部 类 的 时 候 ， 与 使 用 普通 类 没什么 不 同 。 在 这 里 ， 实 际 的 区 
别 只 是 内 部 类 的 名 字 是 伐 套 在 Parcell 里 面 的 。 不 过 你 将 会 看 到 ， 这 并 不 是 唯一 的 区 别 。 

更 典型 的 情况 是 ， 外 部 类 将 有 一 个 方法 ， 访 方法 返回 一 个 指向 内 部 类 的 引用 ， 就 像 在 to0 和 
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contents0 方 法 中 看 到 的 那样 : 


//: innerclasses/Parcel2.java 
// Returning a reference to an inner class. 


public class Parcel2 { 
class Contents { 
private int i = 11; 
public int value() { return i; } 
} 
class Destination { 
private String label; 
Destination(String whereTo) { 
label = whereTo; 
} 
String readLabel() { return label; } 


public Destination to(String s) { 
return new Destination(s) ; 


} 
public Contents contents() { 
return new Contents(); 


} 

public void ship(String dest) { 
Contents c = contents(); 
Destination d = to(dest); 
System. out.printin(d.readLabel()); 

} 

public static void main(String[] args) { 
Parcel2 p = new Parcel2(); 
p.ship("Tasmania"); 
Parcel2 q = new Parcel2(); 
// Defining references to inner classes: 
Parcel2.Contents c = q.contents(); 
Parcel2. Destination d = q.to("Borneo"); 


} Me Output: 

Tasmania 

e011:~ 

如 果 想 从 外 部 类 的 非 静 态 方法 之 外 的 任意 位 置 创建 某 个 内 部 类 的 对 象 ， 那 么 必须 像 在 
main0 方 法 中 那样 ， 具 体 地 指明 这 个 对 象 的 类 型 ，OuterClassName.InnerClassName。 

练习 1:(1) 编写 一 个 名 为 Outer 的 类 ， 它 包含 一 个 名 为 Inner 的 类 。 在 Outer 中 添加 一 个 方法 ， 
它 返 回 一 个 Inner 类 型 的 对 象 。 在 main0 中 ， 创 建 并 初始 化 一 个 指向 某 个 Inner 对 象 的 引用 。 


10.2 链接 到 外 部 类 


到 目前 为 止 ， 内 部 类 似乎 还 只 是 一 种 名 字 隐 藏 和 组 织 代码 的 模式 。 这 些 是 很 有 用 ， 但 还 不 
是 最 引 人 注 目的 ， 它 还 有 其 他 的 用 途 。 当 生成 一 个 内 部 类 的 对 象 时 ， 此 对 象 与 制造 它 的 外 围 对 
% (enclosing object) 之 间 就 有 了 一 种 联系 ， 所 以 它 能 访问 其 外 围 对 象 的 所 有 成 员 ， 而 不 需要 任 
何 特殊 条 件 。 此 外 ， 内 部 类 还 拥有 其 外 围 类 的 所 有 元 素 的 访问 权 。 下 面 的 例子 说 明了 这 点 : 


//: innerclasses/Sequence. java 
// Holds a sequence of Objects. 


interface Selector { 
boolean end(); 
Object current(); 
void next(); 

} 


日 这 与 Cr+ 数 套 类 的 设计 非常 不 同 ,在 Ct+ 中 只 是 单纯 的 名 字 隐 荐 机 制 ， 与 外 围 对 象 没有 联系 ， 也 没有 隐 含 的 访问 权 。 
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public class Sequence { 
private Object[] items; 
private int next = 0; 
public Sequence(int size) { items = new Object[size]; } 
public void add(Object x) { 
if (next < items. Length) 
items[next++] = x; 
} 


Private class SequenceSelector implements Selector { 
private int i = 0; 
Public boolean end() { return i == items.length: } 
public Object current() { return items[i]; } 
public void next() { if(i < items.length) i++; } 


Public Selector selector() { 
return new SequenceSelector(); 


) 
public static void main(String{] args) { 
Sequence sequence = new Sequence(19) ; 
for(int i = 0; i < 10; i++) 
sequence. add (Integer.toString(i)); 
Selector selector = sequence. selector (); 
while(!selector.end()) { 
System.out.print(selector.current() + " "); 
selector .next(); 


) 

} /* Output: 
0123456789 
Whim 


Sequenece 类 只 是 一 个 固定 大 小 的 Object 的 数组 ， 以 类 的 形式 包装 了 起 来 。 可 以 调用 add( 在 
序列 末 增 加 新 的 Object (只 要 还 有 空间 ) 。 要 获取 Sequenee 中 的 每 一 个 对 象 ， 可 以 使 用 Selector 
接口 。 这 是 “和 迭代 器 ”设计 模式 的 一 个 例子 ， 在 本 书 稍 后 的 部 分 将 更 多 地 学 习 它 。Selector 克 许 
你 检查 序列 是 否 到 末尾 了 (end0)， 访 问 当 前 对 象 (currentO)， 以 及 移 到 序列 中 的 下 一 个 对 象 
(next0)。 因 为 Selector 是 一 个 接口 ， 所 以 别 的 类 可 以 按 它们 自己 的 方式 来 实现 这 个 接口 ， 并 且 
另 的 方法 能 以 此 接口 为 参数 ， 来 生成 更 加 通用 的 代码 。 

这 里 ，SequenceSelector 是 提供 Selector 功 能 的 private 类 。 可 以 看 到 ， 在 main0 中 创建 了 一 
个 Sequence， 并 向 其 中 添加 了 一 些 String 对 象 。 然 后 通过 调用 selector0 获 取 一 个 Selector， 并 用 
它 在 Sequence 中 移动 和 选择 每 一 个 元 素 。 

最 初 看 到 SequenceSelector， 可 能 会 觉得 它 只 不 过 是 另 一 个 内 部 类 罢了 。 但 请 仔细 观察 它 ， 
注意 方法 end0、current0 和 next0 都 用 到 了 objects， 这 是 一 个 引用 ， 它 并 不 是 SequenceSelector 
的 一 部 分 ， 而 是 外 围 类 中 的 一 个 private 字 段 。 然 而 内 部 类 可 以 访问 其 外 围 类 的 方法 和 字段 ， 就 
像 自己 拥有 它们 似 的 ， 这 带 来 了 很 大 的 方便 ， 就 如 前 面 的 例子 所 示 。 

所 以 内 部 类 自动 拥有 对 其 外 围 类 所 有 成 员 的 访问 权 。 这 是 如 何 做 到 的 呢 ? 当 某 个 外 围 类 的 
对 象 创建 了 一 个 内 部 类 对 象 时 ， 此 内 部 类 对 象 必定 会 秘密 地 捕获 一 个 指向 那个 外 围 类 对 象 的 引 
用 。 然 后 ， 在 你 访问 此 外 围 类 的 成 员 时 ， 就 是 用 那个 引用 来 选择 外 围 类 的 成 员 。 幸 运 的 是 ， 编 
译 器 会 帮 你 处 理 所 有 的 细节 ， 但 你 现在 可 以 看 到 : 内 部 类 的 对 象 只 能 在 与 其 外 围 类 的 对 象 相关 
联 的 情况 下 才能 被 创建 〈 就 像 你 应 该 看 到 的 ， 在 内 部 类 是 非 static 类 时 ) 。 构 建 内 部 类 对 象 时 ， 
需要 一 个 指向 其 外 围 类 对 象 的 引用 ， 如 果 编 译 器 访问 不 到 这 个 引用 就 会 报错 。 不 过 绝 大 多 数 时 
候 这 都 无 需 程序 员 操 心 。 

练习 2: (1) 创建 一 个 类 ， 它 持 有 一 个 String， 并 且 有 一 个 显示 这 个 String 的 toString( 方 法 。 
将 你 的 新 类 的 若干 个 对 象 添加 到 一 个 Sequence 对 象 中 ， 然 后 显示 它们 。 

练习 3: (1) 修改 练习 1， 使 得 Outer 类 包含 一 个 private String 域 (由 构造 器 初始 化 ) ， 而 
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Inner 包 含 一 个 显示 这 个 域 的 toString0 方 法 。 创 建 一 个 Inner 类 型 的 对 象 并 显示 它 。 
10.3 使 用 .this 与 .new 


如 果 你 需要 生成 对 外 部 类 对 象 的 引用 ， 可 以 使 用 外 部 类 的 名 字 后 面 紧 跟 圆 点 和 this。 这 样 产 
生 的 引用 自动 地 具有 正确 的 类 型 ， 这 一 点 在 编译 期 就 被 知晓 并 受到 检查 ， 因 此 没有 任何 运行 时 
开销 。 下 面 的 示例 展示 了 如 何 使 用 .this: 


/1/: innerclasses/DotThis. java 
// Qualifying access to the outer-class object. 


public class DotThis { 
void f() { System.out.printin("DotThis.f()"); } 
public class Inner { 
public DotThis outer() { 
return DotThis.this; 
1/ A plain "this" would be Inner's “this” 
} 
} 
public Inner inner() { return new Inner(); } 
public static void main(String[) args) { 
DotThis dt = new DotThis(); 
DotThis.Inner dti = dt.inner(); 
dti.outer().#0); 
$ 
} /* Output: 
DotThis. fO 
Whim 


有 时 你 可 能 想 要 告知 某 些 其 他 对 象 ， 去 创建 其 某 个 内 部 类 的 对 象 。 要 实现 此 目的 ， 你 必须 
在 new 表达 式 中 提供 对 其 他 外 部 类 对 象 的 引用 ， 这 是 需要 使 用 .new 语 法， 就 像 下 面 这 样 ; 


//: innerclasses/DotNew. java 
// Creating an inner class directly using the .new syntax. 
public class DotNew { 
public class Inner {} 
public static void main(String(] args) { 
DotNew dn = new DotNew() ; 
DotNew. Inner dni = dn.new Inner(); 


} 
} Mi~ 


要 想 直接 创建 内 部 类 的 对 象 ， 你 不 能 按照 你 想象 的 方式 ， 去 引用 外 部 类 的 名 字 DotNew， 而 
是 必须 使 用 外 部 类 的 对 象 来 创建 该 内 部 类 对 象 ， 就 像 在 上 面 的 程序 中 所 看 到 的 那样 。 这 也 解决 
了 内 部 类 名 字 作用 域 的 问题 ， 因 此 你 不 必 声明 (实际 上 你 不 能 声明 ) dn.new DotNew.Inner0。 

在 拥有 外 部 类 对 象 之 前 是 不 可 能 创建 内 部 类 对 象 的 。 这 是 因为 内 部 类 对 象 会 暗暗 地 连接 到 
创建 它 的 外 部 类 对 象 十 。 但 是 ， 如 果 你 创建 的 是 洗 套 类 (静态 内 部 类 )， 那 么 它 就 不 需要 对 外 部 
类 对 象 的 引用 。 

下 面 你 可 以 看 到 将 .new 应 用 于 Parcel 的 示例 : 

//: innerclasses/Parcel3. java 


// Using .new to create instances of inner classes. 


public class Parcel3 { 

class Contents ( 
private int i = 11; 
public int value() { return i; } 

} 

class Destination { 
private String label; 
Destination(String whereTo) { label = whereTo: } 








由 
回 
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String readLabel() { return label; } 


public static void main(String[] args) { 
Parcel3 p = new Parcel3(); 
// Must use instance of outer class 
// to create an instance of the inner class: 
Parcel3.Contents c = p.new Contents(); 
Parcel3.Destination d = p.new Destination(*Tasmania"); 


} 
} Mh 
练习 4，(2) 在 Sequence.SequenceSelector 类 中 增加 一 个 方法 ， 它 可 以 生成 对 外 部 类 Sequence 
的 引用 。 
练习 5: (1) 创建 一 个 包含 内 部 类 的 类 ， 在 另 一 个 独立 的 类 中 ， 创 建 此 内 部 类 的 实例 。 


10.4 内 部 类 与 向 上 转型 


当 将 内 部 类 向 上 转型 为 其 基 类 ， 尤 其 是 转型 为 一 个 接口 的 时 候 ， 内 部 类 就 有 了 用 武之 地 。 
(从 实现 了 某 个 接口 的 对 象 ， 得 到 对 此 接口 的 引用 ， 与 向 上 转型 为 这 个 对 象 的 基 类 ， 实 质 上 效果 
是 一 样 的 。) 这 是 因为 此 内 部 类 一 一 某 个 接口 的 实现 一 一 能 够 完全 不 可 见 ， 并 且 不 可 用 。 所 得 到 
的 只 是 指向 基 类 或 接口 的 引用 ， 所 以 能 够 很 方便 地 隐藏 实现 细节 。 

我 们 可 以 创建 前 一 个 示例 的 接口 : 


//: innerclasses/Destination. java 

public interface Destination { 
String readLabel(); 

} Ii~ 


41: innerclasses/Contents. java 

public interface Contents { 
int value(); 

) M~ 


现在 Contents 和 Destination 表 示 客 户 端 程序 员 可 用 的 接口 。( 记 住 ， 接 口 的 所 有 成 员 自 动 被 
设置 为 public 的 。) 
当 取得 了 一 个 指向 基 类 或 接口 的 引用 时 ， 甚 至 可 能 无 法 找 出 它 确切 的 类 型 ， 看 下 面 的 例子 ， 


//: innerc1asses/Testparcel.java 


class Parce14 { 
private class PContents implements Contents { 
private int i = 11; 
public int value() { return i; } 


} 
Protected class PDestination implements Destination { 
private String label; 
private PDestination(String whereTo) { 
label = whereTo; 


} 
public String readLabel() { return label: } 


} 

public Destination destination(String s) { 
return new PDestination(s) ; 

} 

public Contents contents() { 
return new PContents(): 


} 


public class TestParcel { 
public static void main(String[] args) { 
Parcel4 p = new Parcel4(); 
Contents c = p.contents(); 
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Destination d = p.destination("Tasmania"); 
// Illegal -- can't access private class: 
//\ Parcel4.PContents pc = p.new PContents(); 


} 
} Mi~ 
Parcel4 中 增加 了 一 些 新 东西 ， 内 部 类 PContents 是 private， 所 以 除了 Parcel4， 没 有 人 能 访 
间 它 。PDestination 是 protected， 所 以 只 有 Parcel4 及 其 子 类 、 还 有 与 Parcel4 同 一 个 包 中 的 类 
(因为 protected 也 给 予 了 包 访问 权 ) 能 访问 PDestination， 其 他 类 都 不 能 访问 PDestination。 这 意 
味 着 ， 如 果 客 户 端 程序 员 想 了 解 或 访问 这 些 成 员 ， 那 是 要 受到 限制 的 。 实 际 上 ， 甚 至 不 能 向 下 
转型 成 private 内 部 类 (或 protected 内 部 类 ， 除 非 是 继承 自 它 的 子 类 ) ， 因 为 不 能 访问 其 名 字 ， 就 
像 在 TestParcel 类 中 看 到 的 那样 。 于 是 ，private 内 部 类 给 类 的 设计 者 提供 了 一 种 途径 ， 通 过 这 种 
方式 可 以 完全 阻止 任何 依赖 于 类 型 的 编码 ， 并 且 完 全 隐藏 了 实现 的 细节 。 此 外 ， 从 客户 端 程序 
员 的 角度 来 看 ， 由 于 不 能 访问 任何 新 增加 的 、 原 本 不 属于 公共 接口 的 方法 ， 所 以 扩展 接口 是 没 
有 价值 的 。 这 也 给 Java 编 译 器 提供 了 生成 更 高 效 代码 的 机 会 。 
练习 6: (2) 在 第 一 个 包 中 创建 一 个 至 少 有 一 个 方法 的 接口 。 然 后 在 第 二 个 包 内 创建 一 个 类 ， 





在 其 中 增加 一 个 protected 的 内 部 类 以 实现 那个 接口 。 在 第 三 个 包 中 ， 继 承 这 个 类 ， 并 在 一 个 方 B53 


法 中 返回 该 protected 内 部 类 的 对 象 ， 在 返回 的 时 候 向 上 转型 为 第 一 个 包 中 的 接口 的 类 型 。 
练习 7: (2) 创建 一 个 含有 private 域 和 private 方 法 的 类 。 创 建 一 个 内 部 类 ， 它 有 一 个 方法 可 
用 来 修改 外 围 类 的 域 ， 并 调用 外 围 类 的 方法 。 在 外 围 类 的 另 一 方法 中 ， 创 建 此 内 部 类 的 对 象 ， 
并 且 调 用 它 的 方法 ， 然 后 说 明 对 外 围 类 对 象 的 影响 。 
练习 8: (2) 确定 外 部 类 是 否 可 以 访问 其 内 部 类 的 private 元 素 。 


10.5 在 方法 和 作用 域内 的 内 部 类 


到 目前 为 止 ， 读 者 所 看 到 的 只 是 内 部 类 的 典型 用 途 。 通 常 ， 如 果 所 读 、 写 的 代码 包含 了 内 
部 类 ， 那 么 它们 都 是 “平凡 的 ”内 部 类 ， 简 单 并 且 容 易 理解 。 然 而 ， 内 部 类 的 语法 覆盖 了 大 量 
其 他 的 更 加 难以 理解 的 技术 。 例 如 ， 可 以 在 一 个 方法 里 面 或 者 在 任意 的 作用 域内 定义 内 部 类 。 
这 么 做 有 两 个 理由 ， 

1) 如 前 所 示 ， 你 实现 了 某 类 型 的 接口 ， 于 是 可 以 创建 并 返回 对 其 的 引用 。 

2) 你 要 解决 一 个 复杂 的 问题 ， 想 创建 一 个 类 来 辅助 你 的 解决 方案 ， 但 是 又 不 希望 这 个 类 是 
公共 可 用 的 。 

在 后 面 的 例子 中 ， 先 前 的 代码 将 被 修改 ， 以 用 来 实现 : 

1) 一 个 定义 在 方法 中 的 类 。 

2) 一 个 定义 在 作用 域内 的 类 ， 此 作用 域 在 方法 的 内 部 。 

3) 一 个 实现 了 接口 的 匿名 类 。 

4) 一 个 匿名 类 ， 它 扩展 了 有 非 默认 构造 器 的 类 。 

5) 一 个 匿名 类 ， 它 执行 字段 初始 化 。 

6) 一 个 匿名 类 ， 它 通过 实例 初始 化 实现 构造 (匿名 类 不 可 能 有 构造 器 )。 

第 一 个 例子 展示 了 在 方法 的 作用 域内 (而 不 是 在 其 他 类 的 作用 域内 ) 创建 一 个 完整 的 类 。 
这 被 称 作 局 部 内 部 类 : 


//: innerclasses/Parcel5.java 
// Nesting a class within a method. 


public class ParcelS { 
public Destination destination(String s) { 
class PDestination implements Destination { 
private String label; 
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private PDestination(String whereTo) { 
label = whereTo; 


} 
public String readLabel() { return label: } 
return new PDestination(s); 


$ 

public static void main(String[] args) { 
Parcel5 p = new Parcel5(); 
Destination d = p.destination("Tasmania"); 


) 

} H~ 

PDestination 类 是 destination() 方 法 的 一 部 分 ， 而 不 是 Parcel5 的 一 部 分 。 所 以 ,在 
destination() 之 外 不 能 访问 PDestination。 注 意 出 现在 return 语 句 中 的 向 上 转型 一 -返回 的 是 
Destination 的 引用 ， 它 是 PDestination 的 基 类 。 当 然 ， 在 destination() 中 定义 了 内 部 类 
PDestination， 并 不 意味 着 一 旦 dest0 方 法 执行 完毕 ，PDestination 就 不 可 用 了 。 

你 可 以 在 同一 个 子 目录 下 的 任意 类 中 对 某 个 内 部 类 使 用 类 标识 符 PDestination， 这 并 不 会 有 
命名 冲突 。 

下 面 的 例子 展示 了 如 何在 任意 的 作用 域内 嵌入 一 个 内 部 类 ; 


//: Annerclasses/Parcel6. java 
17 Nesting a class within a scope. 
public class Parcel6 { 
private void internalTracking(boolean b) { 
ifb) { 
class TrackingSlip { 
private String id; 
TrackingSlip(String s) { 
id = s; 





} 
String getSlip() { return id; } 


} 
TrackingSlip ts = new TrackingSlip("slip"); 
String s = ts.getSlip(); 


// Can't use it here! Out of scope: 
//! TrackingSlip ts = new TrackingSlip(*x"); 


} 
public void track() { internalTracking(true); } 
public static void main(String[] args) { 
Parcel6 p = new Parcel6(); 
p.track(); 


) 

} Ii~ 

TrackingSlip 类 被 修 入 在 站 语句 的 作用 域内 ， 这 并 不 是 说 该 类 的 创建 是 有 条 件 的 ， 它 其 实 与 
别 的 类 一 起 编译 过 了 。 然 而 ， 在 定义 TrackingSlip 的 作用 域 之 外 ， 它 是 不 可 用 的 ， 除 此 之 外 ， 它 
与 普通 的 类 一 样 。 

练习 9: (1) 创建 一 个 至 少 有 一 个 方法 的 接口 。 在 某 个 方法 内 定义 一 个 内 部 类 以 实现 此 接口 ， 
这 个 方法 返回 对 此 接口 的 引用 。 

练习 10: (1) 重复 前 一 个 练习 ， 但 将 内 部 类 定义 在 某 个 方法 的 一 个 作用 域内 。 

练习 11: (2) 创建 一 个 private 内 部 类 ， 让 它 实现 一 个 public 接 口 。 写 一 个 方法 ， 它 返回 一 个 
指向 此 private 内 部 类 的 实例 的 引用 ， 并 将 此 引用 向 上 转型 为 该 接口 类 型 。 通 过 尝试 向 下 转型 ， 
说 明 此 内 部 类 被 完全 隐藏 了 。 


10.6 匿名 内 部 类 
下 面 的 例子 看 起 来 有 点 奇怪 : 
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//: innerclasses/Parcel7.java 
// Returning an instance of an anonymous inner class. 


public class Parcel7 { 
public Contents contents() { 
return new Contents() { // Insert a class definition 
private int i = 11; 
public int value() { return i; } 
}; // Semicolon required in this case 
} 
public static void main(String{] args) { 
Parcel? p = new Parcel7(): 
Contents c = p.contents(); 


} 
} Mi~ 
contents0 方 法 将 返回 值 的 生成 与 表示 这 个 返回 值 的 类 的 定义 结合 在 一 起 ! 另外 ， 这 个 类 是 
匿名 的 ， 它 没有 名 字 。 更 精 的 是 ， 看 起 来 似乎 是 你 正 要 创建 一 个 Contents 对 象 。 但 是 然后 (在 
到 达 语 名 结束 的 分 号 之 前 ) 你 却说 :“ 等 一 等 ， 我 想 在 这 里 插入 一 个 类 的 定义 。” 
这 种 奇怪 的 语法 指 的 是 :“ 创 建 一 个 继承 自 Contents 的 匿名 类 的 对 象 。” 通 过 mew 表达 式 返 回 
的 引用 被 自动 向 上 转型 为 对 Contents 的 引用 。 上 述 匿名 内 部 类 的 语法 是 下 述 形式 的 简化 形式 ， 


/1: innerclasses/Parcel7b. java 
// Expanded version of Parcel7.java 


public class Parcel7b { 
class HyContents implements Contents { 
private int i = 11; 
public int value() { return i; } 
} 
Public Contents contents() { return new MyContents(); 
public static void main(String{] args) { 
Parcel7b p = new Parcel7b(); 
Contents ¢ = p.contents(); 





} 
dh 


在 这 个 匿名 内 部 类 中 ， 使 用 了 默认 的 构造 器 来 生成 Contents。 下 面 的 代码 展示 的 是 ， 如 果 
你 的 基 类 需要 一 个 有 参数 的 构造 器 ， 应 该 怎么 办 : 


//: innerclasses/Parcel8.java 
// Calling the base-class constructor. 


public class Parcels { 
public Wrapping wrapping(int x) { 
// Base constructor call: 


return new Wrapping(x) { // Pass constructor argument. 
public int value() { 
return super.value() * 47; 


} 
}; // Semicolon required 


Public static void main(String[] args) { 
Parcel8 p = new Parcel8(): 
Wrapping w = p.wrapping(18); 


} Mi~ 


| 只 需 简单 地 传递 合适 的 参数 给 基 类 的 构造 器 即 可 ， 这 里 是 将 x 传 进 new Wrapping), RE 
Wrapping 只 是 一 个 具有 具体 实现 的 普通 类 ， 但 它 还 是 被 其 导出 类 当 作 公共 “接口 ”来 使 用 : 


//: innerclasses/Wrapping. java 
public class Wrapping { 
| private int 4; 
Public Wrapping(int x) { i =x; } 
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public int value() { return i; } 
} Mi~ 


你 会 注意 到 ，Wrapping 拥 有 一 个 要 求 传递 一 个 参数 的 构造 器 ， 这 使 得 事情 变 得 更 加 有 趣 了 。 

在 匿名 内 部 类 末尾 的 分 号 ， 并 不 是 用 来 标记 此 内 部 类 结束 的 。 实 际 上 ， 它 标记 的 是 表达 式 
的 结束 ， 只 不 过 这 个 表达 式 正 巧 包含 了 匿名 内 部 类 罢了 。 因 此 ， 这 与 别 的 地 方 使 用 的 分 号 是 一 
致 的 。 

在 匿名 类 中 定义 字段 时 ， 还 能 够 对 其 执行 初始 化 操作 


//: innerclasses/Parcel9. java 
// An anonymous inner class that performs 
// initialization. A briefer version of Parcel5. java. 


public class Parcel9 { 
// Argument must be final to use inside 
// anonymous inner class: 
public Destination destination(final String dest) { 
return new Destination() { 
private String label = dest; 
public String readLabel() { return label; } 
} 
public static void main(String[] args) { 
Parcel9 p = new Parcel9(); 
Destination d = p.destination(*Tasmania*) ; 


} 

) Mi~ 

如 果 定 义 一 个 匿名 内 部 类 ， 并 且 和 希望 它 使 用 一 个 在 其 外 部 定义 的 对 象 ， 那 么 编译 器 会 要 求 
其 参数 引用 是 final 的 ， 就 像 你 在 destination0 的 参数 中 看 到 的 那样 。 如 果 你 忘记 了 ， 将 会 得 到 一 
个 编译 时 错误 消息 。 

如 果 只 是 简单 地 给 一 个 字段 赋值 ， 那 么 此 例 中 的 方法 是 很 好 的 。 但 是 ， 如 果 想 做 一 些 类 似 
构造 器 的 行为 ， 该 怎么 办 呢 ? 在 匿名 类 中 不 可 能 有 命名 构造 器 (因为 它 根 本 没 名 字 ! )， 但 通过 
实例 初始 化 ， 就 能 够 达到 为 匿名 内 部 类 创建 一 个 构造 器 的 效果 ， 就 像 这 样 : 


//: innerclasses/AnonymousConstructor .java 
// Creating a constructor for an anonymous inner class. 
import static net.mindview.util.Print.*; 


abstract class Base { 
public Base(int i) ( 
print("Base constructor, i =" + i): 


} 
public abstract void f(); 


public class AnonymousConstructor { 
public static Base getBase(int 1) { 
return new Base(1) { 
{ print("Inside instance initializer"); } 
public void fO { 
print(*In anonymous ()"); 
} 
}; 
} 
Public static void main(String[] args) { 
Base base = getBase(47); 
base.f(): 


$ 
} /* Output: 
Base constructor, i = 47 
Inside instance initializer 
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In anonymous f() 
Ii~ 


在 此 例 中 ， 不 要 求 变量 i 一 定 是 final 的 。 因 为 i 被 传递 给 匿名 类 的 基 类 的 构造 器 ， 它 并 不 会 在 
匿名 类 内 部 被 直接 使 用 。 

下 例 是 带 实 例 初始 化 的 “parcel” 形 式 。 注 意 destination0 的 参数 必须 是 final 的 ， 因 为 它们 
是 在 匿名 类 内 部 使 用 的 。 


//: nnerclasses/Parcel19.java 
// Using "instance initialization" to perform 
// construction on an anonymous inner class. 





public class Parcel19 { 
public Destination 
destination(final String dest, final float price) { 
return new Destination() { 
private int cost; 
// Instance initialization for each object: 
t 
cost = Math, round(price) ; 
if(cost > 100) 
System. out.printin("Over budget!") ; 


private String label = dest; 

public String readLabel() { return label; } 
k 
} 
public static void main(String() args) { 

Parcel10 p = new Parcel19(); 

Destination d = p.destination("Tasmania”, 161.395F); 


} } Output: 

Over budget! 

“H~ 

在 实例 初始 化 操作 的 内 部 ， 可 以 看 到 有 一 段 代码 ， 它 们 不 能 作为 字段 初始 化 动作 的 一 部 分 
来 执行 (就 是 if 语句 )。 所 以 对 于 匿名 类 而 言 ， 实 例 初始 化 的 实际 效果 就 是 构造 器 。 当 然 它 受到 
了 限制 一 你 不 能 重 载 实例 初始 化 方法 ， 所 以 你 仅 有 一 个 这 样 的 构造 器 。 

匿名 内 部 类 与 正规 的 继承 相 比 有 些 受 限 ， 因 为 匿名 内 部 类 既 可 以 扩展 类 ， 也 可 以 实现 接口 ， 
但 是 不 能 两 者 兼备 。 而 且 如 果 是 实现 接口 ， 也 只 能 实现 一 个 接口 。 

练习 12: (1) 重复 练习 7， 这 次 使 用 匿名 内 部 类 。 

练习 13: (1) 重复 练习 9， 这 次 使 用 匿名 内 部 类 。 

练习 14: (1) 修改 interfaces/HorrorShow.java， 用 匿名 类 实现 DangerousMonster 和 Vampire。 

练习 15，(2) 创建 一 个 类 ， 它 有 非 默认 的 构造 器 ( 即 需 要 参数 的 构造 器 ) ， 并 且 没 有 默认 构造 
器 (没有 无 参数 的 构造 器 )。 创 建 第 二 个 类 ， 它 包含 一 个 方法 ， 能 够 返回 对 第 一 个 类 的 对 象 的 引 
用 。 通 过 写 一 个 继承 自 第 一 个 类 的 匿名 内 部 类 ， 来 创建 一 个 返回 对 象 。 
10.6.1 再 访 工厂 方法 

看 看 在 使 用 匿名 内 部 类 时 ，interfaces/Factoriesjava 示 例 变 得 多 么 美妙 呀 : 


//: innerclasses/Factories.java 
import static net.mindview.util.Print.*; 


interface Service { 
void method1(); 
void method2(); 
a} 
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interface ServiceFactory { 
Service getService(); 
} 
class Implementation] implements Service { 
private Implementation1() {} 
public void method1() {print("Implementation1 method1");} 
public void method2() {print(*Implementation1 method2") ;} 
public static ServiceFactory factory = 
new ServiceFactory() { 
public Service getService() { 
return new Implementation1() ; 
} 
L 
} 


Class Implementation2 implements Service { 
private Implementation2() {} 
public void methodi() {print("Implementation2 method1");} 
public void method2() {print ("Implementation2 method2");} 
public static ServiceFactory factory = 
new ServiceFactory() { 
public Service getService() { 
return new Implementation2(); 
} 
J 


public class Factories { 
public static void serviceConsumer(ServiceFactory fact) { 
Service s = fact.getService(); 
S.method1() ; 
5s.method2(); 
} 
public: static void main(String[] args) { 
serviceConsumer (Implementation1. factory); 
// Implementations are completely interchangeable: 
serviceConsumer (Implementation2. factory) ; 
}》 
} /* Output: 
Implementation] method1 
Implementation] method2 
Implementation2 method1 
Implementation? method2 
Wm 


现在 用 于 Impiementation1 和 Implementation2 的 构造 器 都 可 以 是 private 的 ， 并 且 没 有 任何 必 


要 去 创建 作为 工厂 的 具名 类 。 另 外 ， 你 经 常 只 需要 单一 的 工厂 对 象 ， 因 此 在 本 例 中 它 被 创建 为 
Service 实 现 中 的 一 个 static 域 。 这 样 所 产生 语法 也 更 具有 实际 意 





interfaces/Games.java 示 例 也 可 以 通过 使 用 匿名 内 部 类 来 改进 : 


//: innerclasses/Games . java 
// Using anonymous inner classes with the Game framework. 
import static net.mindview.util.Print.*; 


interface Game { boolean move(); } 
interface GameFactory { Game getGame(): } 


class Checkers implements Game { 

private Checkers() {} 

private int moves = 0; 

private static final int MOVES 

public boolean move() { 
print("Checkers move "+ moves); 
return ++moves != MOVES; 

} 

public static GameFactory factory = new GameFactory() { 
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public Game getGame() { return new Checkers(); } 
} 


class Chess implements Game { 
private Chess() {} 
private int moves = 0: 
private static final int MOVES = 4; 
public boolean move() { 
print("Chess move ”+ moves); 
return +#moves != MOVES; 


} 

public static GameFactory factory = new GameFactory() { 
public Game getGame() { return new Chess(); } 

X 
} 


public class Games { 
public static void playGame(GameFactory factory) { 
Game s = factory. getGame() ; 
whi Le(s.move()) 


} 

public static void main(String[] args) { 
playGame (Checkers. factory) ; 
playGame(Chess. factory) ; 


} 

} /* Output: 
Checkers move @ 
Checkers move 1 
Checkers move 2 
Chess move 9 
Chess move 1 
Chess move 2 
Chess move 3 
Whim 


请 记 住 在 第 9 章 最 后 给 出 的 建议 :优先 使 用 类 而 不 是 接口 。 如 果 你 的 设计 中 需要 某 个 接口 ， 
你 必须 了 解 它 。 否 则 ， 不 到 迫不得已 ， 不 要 将 其 放 到 你 的 设计 中 。 

练习 16: (1) 修改 第 9 章 中 练习 18 的 解决 方案 ， 让 它 使 用 匿名 内 部 类 。 

练习 17: (1) 修改 第 9 章 中 练习 19 的 解决 方案 ， 让 它 使 用 匿名 内 部 类 。 


10.7 REX 


如 果 不 需要 内 部 类 对 象 与 其 外 围 类 对 象 之 间 有 联系 ， 那 么 可 以 将 内 部 类 声明 为 static。 这 通 
常 称 为 央 认 类 。。 想 要 理解 static 应 用 于 内 部 类 时 的 含义 ， 就 必须 记 住 ， 普 通 的 内 部 类 对 象 隐 式 
地 保存 了 一 个 引用 ， 指 向 创建 它 的 外 围 类 对 象 。 然 而 ， 当 内 部 类 是 static 的 时 ， 就 不 是 这 样 了 。 
ERR 

1) 要 创建 找 套 类 的 对 象 ， 并 不 需要 其 外 围 类 的 对 象 。 

2) 不 能 从 网 套 类 的 对 象 中 访问 非 静态 的 外 围 类 对 象 。 

媒 套 类 与 普通 的 内 部 类 还 有 一 个 区 别 。 普 通 内 部 类 的 字段 与 方法 ， 只 能 放 在 类 的 外 部 层次 
上 ， 所 以 普通 的 内 部 类 不 能 有 static 数 据 和 static 字 段 ， 也 不 能 包含 嵌 套 类 。 但 是 网 套 类 可 以 包含 
所 有 这 些 东西 : 


//: innerclasses/Parcel11. java 
// Nested classes (static inner classes). 


public class Parcelll { 


日 SCHHRRA KEL, RPE PARTE LARA, M Eavah FEL, 
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private static class ParcelContents implements Contents { 
private int i = 11; 
public int value() { return i; } 


} 
protected static class ParcelDestination 
implements Destination { 
private String label; 
private ParcelDestination(String whereTo) { 
label = whereTo; 


public String readLabel() { return label; } 
// Nested classes can contain other static elements: 
public static void fO {} 
static int x = 10; 
static class AnotherLevel { 
public static void fO {} 
static int x = 10; 
) 


} 
public static Destination destination(String s) { 
return new ParcelDestination(s); 


} 
public static Contents contents() { 
return new ParcelContents(); 


} 

public static void main(String[] args) { 
Contents c = contents (): 
Destination d = destination("Tasmania"); 


} 
} Mi~ 


在 main0 中 ， 没 有 任何 Parcell1 的 对 象 是 必需 的 ， 而 是 使 用 选取 static 成 员 的 普通 语法 来 调用 
方法 一 一 这 些 方法 返回 对 Contents 和 Destination 的 引用 。 

就 像 你 在 本 章 前 面 看 到 的 那样 ， 在 一 个 普通 的 〈 非 static) 内 部 类 中 ， 通 过 一 个 特殊 的 this 
引用 可 以 链接 到 其 外 围 类 对 象 。 嵌 套 类 就 没有 这 个 特殊 的 this 引 用 ， 这 使 得 它 类 似 于 一 个 static 
方法 。 

练习 18: (1) 创建 一 个 包含 做 套 类 的 类 。 在 main0 中 创建 其 内 部 类 的 实例 。 

练习 19: (2) 创建 一 个 包含 了 内 部 类 的 类 ， 而 此 内 部 类 又 包含 有 内 部 类 。 使 用 嵌 套 类 重复 这 
个 过 程 。 注 意 编译 器 生成 的 .class 文 件 的 名 字 。 

10.7.1 接口 内 部 的 类 

正常 情况 下 ， 不 能 在 接口 内 部 放置 任何 代码 ， 但 嵌 套 类 可 以 作为 接口 的 一 部 分 。 你 放 到 接 
口中 的 任何 类 都 自动 地 是 public 和 static 的 。 因 为 类 是 static 的 ， 只 是 将 嵌 套 类 置 于 接口 的 命名 空 
间 内 ， 这 并 不 违反 接口 的 规则 。 你 甚至 可 以 在 内 部 类 中 实现 其 外 围 接口 ， 就 像 下 面 这 样 : 


//: innerclasses/ClassInInterface.java 
// {main: ClassInInterface$Test} 


public interface ClassInInterface { 
void howdy(); 
class Test implements ClassInInterface { 
public void howdy() { 
‘System. out. printin("Howdy!"); 
} 
public static void main(String[] args) { 
new Test().howdy(); 
} 


} 
} /* Output: 
Howdy! 
Y~ 





| 
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如 果 你 想 要 创建 某 些 公共 代码 ， 使 得 EAT AERTS HAAR SOSH 那么 使 
用 接口 内 部 的 霸 套 类 会 显得 很 方便 。 

我 曾 在 本 书 中 建议 过 ， 在 每 个 类 中 都 写 一 个 main0 方 法 ， 用 来 测试 这 个 类 。 这 样 做 有 一 个 
缺点 ， 那 就 是 必须 带 着 那些 已 编译 过 的 额外 代码 。 如 果 这 对 你 是 个 麻烦 ， 那 就 可 以 使 用 伐 套 类 
来 放置 测试 代码 。 


//: innerclasses/TestBed. java 
// Putting test code in a nested class. 
// {main: TestBedSTester) 


public class TestBed { 
public void f() { System.out.printin("f()"); } 
public static class Tester { 
public static void main(String(] args) { 
TestBed t = new TestBed(); 
tTO: 
} 
} 
} /* Output: 
t0 
I~ 


这 生成 了 一 个 独立 的 类 TestBed$Tester (要 运行 这 个 程序 ， 执 行 java TestBed$Tester 即 可 ， 
在 Unix/Linux 系 统 中 必须 转 义 $)。 可 以 使 用 这 个 类 来 做 测试 ， 但 是 不 必 在 发 布 的 产品 中 包含 它 ， 
在 将 产品 打包 前 可 以 简单 地 删除 TestBed$Tester.class。 
练习 20: (1) 创建 一 个 包含 伐 套 类 的 接口 ， 实 现 此 接口 并 创建 供 套 类 的 实例 。 
练习 21: (2) 创建 一 个 包含 供 套 类 的 接口 ， 该 做 套 类 中 有 一 个 static 方 法 ， 它 将 调用 接口 中 
的 方法 并 显示 结果 。 实 现 这 个 接口 ， 并 将 这 个 实现 的 一 个 实例 传递 给 这 个 方法 。 36 
10.7.2 从 多 层 嵌 套 类 中 访问 外 部 类 的 成 员 
一 个 内 部 类 被 供 套 多 少 层 并 不 重要 一 一 它 能 透明 地 访问 所 有 它 所 供 入 的 外 围 类 的 所 有 成 
员 ， 如 下 所 示 : 


3 


/1/: innerclasses/MultiNestingAccess. java 
// Nested classes can access all members of all 
// levels of the classes they are nested within. 


class MNA ( 
| private void 10 {} 
class A { 
| private void g() {} 
public class B { 
i void h() { 
l BO: 
f0: 
} 
} 
} 


public class HultiNestingAccess { 
public static void main(String[] args) { 
MNA mna = new MNA(); 
MNA.A mnaa = mna.new A(); 
HNA.A.B mnaab = mnaa.new B(); 


mnaab.h() 
} :~ 
日 再 次 感谢 Martin Danner。 
| 
| 
1 
1 


| 
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可 以 看 到 在 MNA.A.B 中 ， 调 用 方法 g80 和 f0 不 需要 任何 条 件 〈 即 使 它们 被 定义 为 private) 。 
这 个 例子 同时 展示 了 如 何 从 不 同 的 类 里 创建 多 层 嵌 套 的 内 部 类 对 象 的 基本 语法 。“.new” 语 法 能 
产生 正确 的 作用 域 ， 所 以 不 必 在 调用 构造 器 时 限定 类 名 。 


10.8 为 什么 需要 内 部 类 


至 此 ,我 们 已 经 看 到 了 许多 描述 内 部 类 的 语法 和 语义 ， 但 是 这 并 不 能 回答 “为 什么 需要 内 
部 类 ”这 个 问题 。 那 么 ， Sun 公 司 为 什么 会 如 此 费心 地 增加 这 项 基本 的 语言 特性 呢 ? 

一 般 说 来 ， 内 部 类 继承 自 某 个 类 或 实现 某 个 接口 ， 内 部 类 的 代码 操作 创建 它 的 外 围 类 的 对 
象 。 所 以 可 以 认为 内 部 类 提供 了 某 种 进入 其 外 围 类 的 窗口 。 

内 部 类 必须 要 回答 的 一 个 问题 是 : 如 果 只 是 需要 一 个 对 接口 的 引用 ， 为 什么 不 通过 外 围 类 
实现 那个 接口 呢 ? 答案 是 :“ 如 果 这 能 满足 需求 ， 那 么 就 应 该 这 样 做 。” 那 么 内 部 类 实现 一 个 接 
口 与 外 围 类 实现 这 个 接口 有 什么 区 别 呢 ? 答案 是 : 后 者 不 是 总 能 享用 到 接口 带 来 的 方便 ， 有 时 
需要 用 到 接口 的 实现 。 所 以 ， 使 用 内 部 类 最 吸引 人 的 原因 是 : 

每 个 内 部 类 都 能 独立 地 继承 自 一 个 (接口 的 ) 实现 ， 所 以 无 论 外 围 类 是 否 已 经 继承 了 某 个 
(接口 的 ) 实现 ， 对 于 内 部 类 都 没有 影响 。 

如 果 没 有 内 部 类 提供 的 、 可 以 继承 多 个 具体 的 或 抽象 的 类 的 能 力 ， 一 些 设计 与 编程 问题 就 
很 难 解决 。 从 这 个 角度 看 ， 内 部 类 使 得 多 重 继承 的 解决 方案 变 得 完整 。 接 口 解决 了 部 分 问题 ， 
而 内 部 类 有 效 地 实现 了 “多 重 继承 ”"。 也 就 是 说 ， 内 部 类 允许 继承 多 个 非 接口 类 型 译注， 类 或 
抽象 类 )。 

为 了 看 到 更 多 的 细节 ， 让 我 们 考虑 这 样 一 种 情形 ; 即 必须 在 一 个 类 中 以 某 种 方式 实现 两 个 
接口 。 由 于 接口 的 灵活 性 ， 你 有 两 种 选择 : 使 用 单一 类 ， 或 者 使 用 内 部 类 ; 


1/: innerctasses/WuttiInterfaces.java 
// Two ways that a class can implement multiple interfaces. 
package innerclasses; 


interface A {} 
interface B {} 


class X implements A, B {} 


class Y implements A { 
B makeB() { 
17 Anonymous inner class: 
return new B() {}; 
} 
} 


public class MultiInterfaces { 

static void takesA(A a) {} 

static void takesB(B b) {} 

public static void main(String{] args) { 
X x = new XO); 
Y y = new YQ): 
takesA(x); 
takesA(y); 
takesB(x) ; 
takesB(y.makeB()) ; 


} 
dh 
当然 ， 这 里 假设 在 两 种 方式 下 的 代码 结构 都 确实 有 有 还 辑 意义 。 然 而 遇 到 问题 的 时 候 ， 通 常 
问题 本 身 就 能 给 出 某 些 指引 ， 告 诉 你 是 应 该 使 用 单一 类 ， 还 是 使 用 内 部 类 。 但 如 果 没 有 任何 其 
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他 限制 ， 从 实现 的 观点 来 看 ， 前 面 的 例子 并 没有 什么 区 别 ， 它 们 都 能 正常 运作 。 
如 果 拥 有 的 是 抽象 的 类 或 具体 的 类 ， 而 不 是 接口 ， 那 就 只 能 使 用 内 部 类 才能 实现 多 重 继承 。 


//: innerclasses/MultiImplementation. java 
// With concrete or abstract classes, inner 

// classes are the only way to produce the effect 
// of “multiple implementation inheritance.” 
package innerclasses; 


class D {} 
abstract class E {} 
class Z extends D { 
E makeE() { return new EO {}; } 
} 


public class Multilmplementation { 
static void takesD(D d) {} 
static void takesE(E e) {} 
public static void main(String[] args) { 
Zz = new Z(); 
takesD(z); 
takesE(z.makeE()); 


) 

} H~ 

如 果 不 需 要 解决 “多 重 继承 ”的 问题 ， 那 么 自然 可 以 用 别 的 方式 编码 ， 而 不 需要 使 用 内 部 
类 。 但 如 果 使 用 内 部 类 ， 还 可 以 获得 其 他 一 些 特性 : 

D 内 部 类 可 以 有 多 个 实例 ， 每 个 实例 都 有 自己 的 状态 信息 ， 并 且 与 其 外 围 类 对 象 的 信息 相 
互 独立 。 

2) 在 单个 外 围 类 中 ， 可 以 让 多 个 内 部 类 以 不 同 的 方式 实现 同一 个 接口 ， 或 继承 同一 个 类 。 
稍 后 就 会 展示 一 个 这 样 的 例子 。 

3) 创建 内 部 类 对 象 的 时 刻 并 不 依赖 于 外 围 类 对 象 的 创建 。 

4) 内 部 类 并 没有 令 人 迷惑 的 “is-a” 关 系 ， 它 就 是 一 个 独立 的 实体 。 

举 个 例子 ， 如 果 Sequencejava 不 使 用 内 部 类 ， 就 必须 声明 “Sequence 是 一 个 Selector”， 对 
于 某 个 特定 的 Sequence 只 能 有 一 个 Selector。 然 而 使 用 内 部 类 很 容易 就 能 拥有 另 一 个 方法 
reverseSelector0， 用 它 来 生成 一 个 反方 向 遍历 序列 的 Selector。 只 有 内 部 类 才 有 这 种 灵活 性 。 

练习 22: (2) 实现 Sequence.java 中 的 reverseSelector0 方 法 。 

练习 23: (4) 创建 一 个 接口 U， 它 包含 三 个 方法 。 创 建 第 一 个 类 A， 它 包含 一 个 方法 ， 在 此 
方法 中 通过 创建 一 个 匿名 内 部 类 ， 来 生成 指向 U 的 引用 。 创 建 第 二 个 类 B， 它 包含 一 个 由 0 构成 
的 数组 。B 应 该 有 几 个 方法 ， 第 一 个 方法 可 以 接受 对 0 的 引用 并 存储 到 数组 中 ， 第 二 方法 将 数组 
中 的 引用 设 为 null， 第 三 个 方法 遍历 此 数组 ， 并 在 U 中 调用 这 些 方法 。 在 main0 中 ， 创 建 一 组 人 A 
的 对 象 和 一 个 B 的 对 象 。 用 那些 A 类 对 象 所 产生 的 U 类 型 的 引用 填充 B 对 象 的 数组 。 使 用 B 回 调 所 
有 A 的 对 象 。 再 从 B 中 移 除 某 些 U 的 引用 。 
10.8.1 闭 包 与 回调 

Me (closure) 是 一 个 可 调用 的 对 象 ， 它 记录 了 一 些 信息 ， 这 些 信息 来 自 于 创建 它 的 作用 
域 。 通 过 这 个 定义 ， 可 以 看 出 内 部 类 是 面向 对 象 的 闭 包 ， 因 为 它 不 仅 包含 外 围 类 对 象 (创建 内 
部 类 的 作用 域 ) 的 信息 ， 还 自动 拥有 一 个 指向 此 外 围 类 对 象 的 引用 ， 在 此 作用 域内 ， 内 部 类 有 
权 操 作 所 有 的 成 员 ， 包 括 private 成 员 。 。 ， 

Java 最 引 人 争 议 的 问题 之 一 就 是 ， 人 们 认为 Java 应 该 包含 某 种 类 似 指针 的 机 制 ， 以 允许 回调 
(callback)。 通 过 回调 ， 对 象 能 够 携带 一 些 信息 ， 这 些 信息 允许 它 在 稍 后 的 某 个 时 刻 调用 初始 的 


pa 
3 
> 








| -- 











372| 








206 # 10% 





对 象 。 稍 后 将 会 看 到 这 是 一 个 非常 有 用 的 概念 。 如 果 回调 是 通过 指针 实现 的 ， 那 么 就 只 能 寄 希 
望 于 程序 员 不 会 误 用 该 指针 。 然 而 ， 读 者 应 该 已 经 了 解 到 ，Java 更 小 心 仔细 ， 所 以 没有 在 语言 
中 包括 指针 。 


通过 内 部 类 提供 闭 包 的 功能 是 优良 的 解决 方案 ， 它 比 指针 更 灵活 、 更 安全 。 见 下 例 ; 


//: innerclasses/Callbacks. java 

// Using inner classes for callbacks 
package innerclasses; 

import static net.mindview.util.Print.*; 


interface Incrementable { 
void increment () ; 


} 


// Very simple to just implement the interface: 
class Calleel implements Incrementable { 
private int i = 
public void incr 
itt; 
print(i); 








} 
} 


class MyIncrement ( 
public void increment() { print("Other operation"); } 
static void f(MyIncrement mi) { mi.increment(); } 


} 


// If your class must implement increment() in 
// some other way, you must use an inner class: 
class Callee2 extends MyIncrement { 
private int i = 0; 
public void increment() { 
super. increment(); 
itt; 
print(i); 
} 
private class Closure implements Incrementable { 
public void increment() { 
// Specify outer-class method, otherwise 
// you'd get an infinite recursion: 
Callee2. this. increment(); 
} 


} 

Incrementable getCallbackReference() { 
return new Closure(); 

} 


class Caller { 
private Incrementable callbackReference; 
Caller(Incrementable cbh) { callbackReference = cbh: } 
void go() { callbackReference. increment(): } 

} 


public class Callbacks { 
public static void main(String{] args) { 

Calleel cl = new Calleel(); 
Callee2 c2 = new Callee2(); 
MyIncrement.f(c2); 
Caller callerl = new Caller(cl); 
Caller caller2 = new Caller (c2.getCallbackReference()); 
callerl.go(); 
callert.go(): 
caller2.go(); 
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caller2.go(); 


} 
} /* Output: 
Other operation 
1 


1 
2 
Other operation 
2 


Other operation 

3 

Sin 

这 个 例子 进一步 展示 了 外 围 类 实现 一 个 接口 与 内 部 类 实现 此 接口 之 间 的 区 别 。 就 代码 而 言 ， 
Calleel 是 简单 的 解决 方式 。Callee2 继 承 自 MyIncrement， 后 者 已 经 有 了 一 个 不 同 的 increment() 
方法 ， 并 且 与 Inerementable 接 口 期 望 的 increment0 方 法 完全 不 相关 。 所 以 如 果 Callee2 继 承 了 
MyIncrement， 就 不 能 为 了 Incrementable 的 用 途 而 覆盖 inerement0 方 法 ， 于 是 只 能 使 用 内 部 类 
独立 地 实现 Incrementable。 还 要 注意 ， 当 创建 了 一 个 内 部 类 时 ， 并 没有 在 外 围 类 的 接口 中 添加 
东西 ， 也 没有 修改 外 围 类 的 接口 。 

注意 ， 在 Callee2 中 除了 getCallbackReference0 以 外 ， 其 他 成 员 都 是 private 的 。 要 想 建立 与 
外 部 世界 的 任何 连接 ，interface Incrementable 都 是 必需 的 。 在 这 里 可 以 看 到 ，interface 是 如 何 
允许 接口 与 接口 的 实现 完全 独立 的 。 

内 部 类 Closure 实 现 了 Incrementabie， 以 提供 一 个 返回 Cailee2 的 “钩子 ”(hook) 一 一 而 且 
是 一 个 安全 的 钩子 。 无 论 谁 获得 此 Inerementable 的 引用 ， 都 只 能 调用 incrementO， 除 此 之 外 没 
有 其 他 功能 (不 像 指针 那样 ， 允 许 你 做 很 多 事情 )。 

Caller 的 构造 器 需要 一 个 Incrementable 的 引用 作为 参数 (虽然 可 以 在 任意 时 刻 捕获 回调 引 
用 )， 然 后 在 以 后 的 某 个 时 刻 ，Caller 对 象 可 以 使 用 此 引用 回调 Callee 类 。 

回调 的 价值 在 于 它 的 灵活 性 一 一 可 以 在 运行 时 动态 地 决定 需要 调用 什么 方法 。 这 样 做 的 好 
处 在 第 22 章 可 以 看 得 更 明显 ， 在 那里 实现 GUI 功能 的 时 候 ， 到 处 都 用 到 了 回调 。 

10.8.2 内 部 类 与 控制 框架 

在 将 要 介绍 的 控制 框架 (control framework) 中 ， 可 以 看 到 更 多 使 用 内 部 类 的 具体 例子 。 

应 用 程序 框架 (application framework) 就 是 被 设计 用 以 解决 某 类 特定 问题 的 一 个 类 或 一 组 
类 。 要 运用 某 个 应 用 程序 框架 ， 通 常 是 继承 一 个 或 多 个 类 ， 并 政 盖 某 些 方法 。 在 覆盖 后 的 方法 
中 ， 编 写 代码 定制 应 用 程序 框架 提供 的 通用 解决 方案 ， 以 解决 你 的 特定 问题 (这 是 设计 模式 中 
模板 方法 的 一 个 例子 (参考 Wwww.MindVeiw.net 上 的 《Thinking in Patterns (with Java)》) 。 模 板 方 
法 包含 算法 的 基本 结构 ， 并 且 会 调用 一 个 或 多 个 可 覆盖 的 方法 ， 以 完成 算法 的 动作 。 设 计 模 式 
总 是 将 变化 的 事物 与 保持 不 变 的 事物 分 离开 ， 在 这 个 模式 中 ， 模 板 方法 是 保持 不 变 的 事物 ， 而 
可 得 盖 的 方 滋 就 是 变化 的 事物 。 

控制 框架 是 一 类 特殊 的 应 用 程序 框架 ， 它 用 来 解决 响应 事件 的 需求 。 主 要 用 来 响应 事件 的 
系统 被 称 作 事 件 驱动 系统 。 应 用 程序 设计 中 常见 的 问题 之 一 是 图 形 用 户 接口 (GUI) ， 它 几乎 完 
全 是 事件 驱动 的 系统 。 在 第 22 章 将 会 看 到 ，Java Swing 库 就 是 一 个 控制 框架 ， 它 优雅 地 解决 了 
GUI 的 问题 ， 并 使 用 了 大 量 的 内 部 类 。 

要 理解 内 部 类 是 如 何 允 许 简单 的 创建 过 程 以 及 如 何 使 用 控制 框架 的 ， 请 考虑 这 样 一 个 控制 
框架 ， 它 的 工作 就 是 在 事件 “就 结 ” 的 时 候 执行 事件 。 虽 然 “ 就 绪 ” 可 以 指 任何 事 ， 但 在 本 例 
中 是 指 基于 时 间 触 发 的 事件 。 接 下 来 的 问题 就 是 ， 对 于 要 控制 什么 ， 控 制 框架 并 不 包含 任何 具 
体 的 信息 。 那 些 信息 是 在 实现 算法 的 action0 部 分 时 ， 通 过 继承 来 提供 的 。 
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首先 ， 接 口 描述 了 要 控制 的 事件 。 因 为 其 默认 的 行为 是 基于 时 间 去 执行 控制 ， 所 以 使 用 抽 
象 类 代替 实际 的 接口 。 下 面 的 例子 包含 了 某 些 实现 : 


//: innerclasses/controller/Event. java 
// The common methods for any control event. 
package innerclasses.controller; 
public abstract class Event { 
private long eventTime; 
protected final long delayTime; 
public Event(long delayTime) { 
this.delayTime = delayTime; 
start(); 


public, void start() { // Allows restarting 
eventTime = System.nanoTime() + delayTime; 


} 
public boolean ready() { 
return System.nanoTime() >= eventTime; 


} 
public abstract void action(); 
} Mi~ 


当 希 望 运 行 Eveat 并 随后 调用 start0 时 ， 那 么 构造 器 就 会 捕获 (从 对 象 创建 的 时 刻 开始 的 ) 
时 间 ， 此 时 间 是 这 样 得 来 的 : start0 获 取 当 前 时 间 ， 然 后 加 上 一 个 延迟 时 间 ， 这 样 生成 触发 事件 
的 时 间 。start0 是 一 个 独立 的 方法 ， 而 没有 包含 在 构造 器 内 ， 因 为 这 样 就 可 以 在 事件 运行 以 后 重 
新 启动 计时 器 ， 也 就 是 能 够 重复 使 用 Event 对 象 。 例 如 ， 如 果 想 要 重复 一 个 事件 ， 只 需 简单 地 在 
action0 中 调用 start( 方 法 。 

ready0 告 诉 你 何 时 可 以 运行 action0 方 法 了 。 当 然 ， 可 以 在 导出 类 中 覆盖 ready0 方 法 ， 使 得 
Event 能 够 基于 时 间 以 外 的 其 他 因素 而 触发 。 

下 面 的 文件 包含 了 一 个 用 来 管理 并 触发 事件 的 实际 控制 框架 。Event 对 象 被 保存 在 
List<Event> 类 型 ( 读 作 “Event 的 列表 ") 的 容器 对 象 中 ， 容 器 会 在 第 11 章 中 详细 介绍 。 目 前 读 
者 只 需要 知道 add( 方 法 用 来 将 一 个 Object 添加 到 List 的 尾 端 ，size0 方 法 用 来 得 到 List 中 元 素 的 个 
数 ，foreach 语 法 用 来 连续 获 联 List 中 的 Event，remove0 方 法 用 来 从 List 中 移 除 指定 的 Event。 


//: innerclasses/controller/Controller .java 
// The reusable framework for control systems. 
Package innerclasses.controller; 

import java.util.*; 


public class Controller { 
17 A class from java.util to hold Event objects: 
private List<Event> eventList = new ArrayList<Event>(); 
Public void addEvent (Event c) { eventList.add(c); } 
public void run() { 
while(eventList.size() > 0) 
// Make a copy so you're not modifying the list ‘ 
// while you're selecting the elements in it: 
for(Event e : new ArrayList<Event>(eventList)) 
if(e.ready()) { > 
System.out.println(e); 
e.action(); 
eventList.remove(e); 


} 
} 
} Ii~ 
run0 方 法 循环 遍历 eventList， 寻 找 就 绪 的 (ready0)、 要 运行 的 Event 对 象 。 对 找到 的 每 一 
个 就 绪 的 (readyO) 事件 ， 使 用 对 象 的 toString0 打 印 其 信息 ， 调 用 其 action0 方 法 ， 然 后 从 队列 
中 移 除 此 Event。 


io - s 
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注意 ， 在 目前 的 设计 中 你 并 不 知道 Event 到 底 做 了 什么 。 这 正 是 此 设计 的 关键 所 在 ,“ 使 变 
化 的 事物 与 不 变 的 事物 相互 分 离 "。 用 我 的 话说 ,“ 变 化 向 量 ”就 是 各 种 不 同 的 Event 对 象 所 具有 
的 不 同行 为 ， 而 你 通过 创建 不 同 的 Event 子 类 来 表现 不 同 的 行为 。 

这 正 是 内 部 类 要 做 的 事情 ， 内 部 类 允许 : 

1) 控制 框架 的 完整 实现 是 由 单个 的 类 创建 的 ， 从 而 使 得 实现 的 细节 被 封装 了 起 来 。 内 部 类 
用 来 表示 解决 问题 所 必需 的 各 种 不 同 的 action0。 

2) 内 部 类 能 够 很 容易 地 访问 外 围 类 的 任意 成 员 ， 所 以 可 以 避免 这 种 实现 变 得 笨拙 。 如 果 没 
有 这 种 能 力 ， 代 码 将 变 得 令 人 讨厌 ， 以 至 于 你 肯定 会 选择 别 的 方法 。 

考虑 此 控制 框架 的 一 个 特定 实现 ， 如 控制 温室 的 运作 。: 控制 灯光 、 水 、 温 度 调节 器 的 开关 ， 
以 及 响 铃 和 重新 启动 系统 ， 每 个 行为 都 是 完全 不 同 的 。 控 制 框架 的 设计 使 得 分 离 这 些 不 同 的 代 
码 变 得 非常 容易 。 使 用 内 部 类 ， 可 以 在 单一 的 类 里 面 产生 对 同一 个 基 类 Event 的 多 种 导出 版 本 。 
对 于 温室 系统 的 每 一 种 行为 ， 都 继承 一 个 新 的 Event 内 部 类 ， 并 在 要 实现 的 action0 中 编写 控制 
代码 。 

作为 典型 的 应 用 程序 框架 ，GreenhouseControls 类 继承 自 Controller: 


//: tnnerclasses/GreenhouseControls . java 

// This produces a specific application of the 
// control system, all in a single class. Inner 
11 classes allow you to encapsulate different 
// functionality for each type of event. 

import innerclasses.controller.*; 


public class GreenhouseControls extends Controller { 
private boolean light = false; 
public class LightOn extends Event { 
public LightOn(long delayTime) { super(delayTime); } 
public void action() { 
// Put hardware control code here to 
// physically turn on the light. 
Light = true; 


} 
public String toString() { return "Light is on"; } 


public class LightOff extends Event { 
public LightOff(long delayTime) { super(delayTime); } 
public void action() { 
// Put hardware control code here to 
// physically turn off the light. 
light = false; 


} 
public String toString() { return “Light is off"; } 


private boolean water = false: 
public class WaterOn extends Event { 
public WaterOn(long delayTime) { super(delayTime); } 
public void action() { 
// Put hardware control code here. 
water = true; 


} 
public String tostring() { 
return "Greenhouse water is on"; 


} 


} 
public class WaterOff extends Event { 


O 基于 某 种 原因 ， 我 一 直 很 乐意 解决 这 个 问题 ， 该 问题 摘自 我 以 前 的 节 《C++ Inside & Out》， 但 是 java 提供 了 更 
优雅 的 解决 方案 。 
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public WaterOff(long delayTime) { super(delayTime); } 
public void action() { 
// Put hardware control code here. 
water = false; 
} 
public String toString() { 
return “Greenhouse water is off"; 
} 
} 
private String thermostat = "Day"; 
public class ThermostatNight extends Event { 
public ThermostatNight(long delayTime) { 
super (deLayTime) ; 
} 
Public void action() { 
// Put hardware control code here. 
thermostat = "Night"; 
} 
public String toString() { 
return "Thermostat on night setting": 
} 
} 
public class ThermostatDay extends Event { 
public ThermostatDay(long delayTime) { 
super (delayTime) ; 
} 
public void action() { 
// Put hardware control code here. 
thermostat = "Day"; 
} 
public String toString() { 
return "Thermostat on day setting 
} 
} 





. // An example of an action() that inserts a 


// new one of itself into the event list: 
public class Bell extends Event { 
public Bell(long delayTime) { super(delayTime); } 
public void action() { 
addEvent (new Bell (delayTime)) ; 
} 
public String toString() { return “Bing!"; } 
} 
public class Restart extends Event { 
private Event[] eventList; 
public Restart(long delayTime, Event[] eventList) { 
super (delayTime) ; 
this.eventList = eventLlist; 
for(Event e : eventList) 
addEvent (e) ; 
} 
public void action() { 
for(Event e : eventList) { 
e.start(); // Rerun each event 
addEvent (e); 
} 
start(); // Rerun this Event 
addEvent (this); 
} 
public String toString() { 
return "Restarting system"; 
} 
} 
public static class Terminate extends Event { 
public Terminate(long delayTime) { super(delayTime); } 
public void action() { System.exit(@): } 
public String toString() { return "Terminating" 
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} 

yh 

注意 ，light、water 和 thermostat 都 属于 外 围 类 GreenhouseControls， 而 这 些 内 部 类 能 够 
自由 地 访问 那些 字段 ， 无 需 限定 条 件 或 特殊 许可 。 而 且 ，action0 方 法 通常 都 涉及 对 某 种 硬件 的 
控制 。 

大 多 数 Event 类 看 起 来 都 很 相似 ， 但 是 Bell 和 Restart 则 比较 特别 。Bell 控 制 响 铃 ， 然 后 在 事 
件 列表 中 增加 一 个 Bell 对 象 ， 于 是 过 一 会 儿 它 可 以 再 次 响 铃 。 读 者 可 能 注意 到 了 内 部 类 是 多 么 像 
多 重 继承 ，Bell 和 Restart 有 Event 的 所 有 方法 ， 并 且 似 乎 也 拥有 GreenhouseContrlos 的 所 
有 方法 。 

一 个 由 Event 对 象 组 成 的 数组 被 递交 给 Restart， 该 数组 要 加 到 控制 器 上 。 由 于 Restart0 也 是 
一 个 Event 对 象 ， 所 以 同样 可 以 将 Restart 对 象 添 加 到 Restartaction0 中 ， 以 使 系统 能 够 有 规律 地 
重新 启动 自己 。 

下 面 的 类 通过 创建 一 个 GreenhouseControls 对 象 ， 并 添加 各 种 不 同 的 Event 对 象 来 配置 该 系 
统 。 这 是 命令 设计 模式 的 一 个 例子 在 eventList 中 的 每 一 个 被 封装 成 对 象 的 请 求 : 


//: innerclasses/GreenhouseController .java 
// Configure and execute the greenhouse system. 
// {Args: 5000} 

‘import innerclasses.controller.*; 





public class GreenhouseController { 
public static void main(String(] args) { 

GreenhouseControls gc = new GreenhouseControls(); 

// Instead of hard-wiring, you could parse 

// configuration information from a text file here: 

Bc.addEvent (gc.new Bell (988)) ; 

Event[] eventList = { 

gc.new ThermostatNight(@), 
gc.new LightOn(209), 
Bc.new LightOff (490), 
gc.new WaterOn (608) , 
gc.new WaterOff (890), 
gc.new ThermostatDay (1400) 

}; 
gc. addEvent (gc.new Restart (2000, eventList)); 
if(args.length == 1) ° 

gc. addEvent ( 

new GreenhouseControls.Terminate( 

new Integer (args{@]))): 
gc.run(); 


} 
} /* Output: 
Bing! 
Thermostat on night setting 
Light is on 
Light is off 
Greenhouse water is on 
Greenhouse water is off 
Thermostat on day setting 
Restarting system 
Términating 
“701:~ 


这 个 类 的 作用 是 初始 化 系统 ， 所 以 它 添加 了 所 有 相应 的 事件 。Restart 事 件 反复 运行 ， 而 且 
它 每 次 都 会 将 eventList 加 载 到 GreenhouseControls 对 象 中 。 如 果 提供 了 命令 行 参数 ， 系 统 会 以 它 
作为 毫秒 数 ， 决 定 什么 时 修 终止 程序 (这 是 测试 程序 时 使 用 的 )。 当 然 ， 更 灵活 的 方法 是 避免 对 
事件 进行 硬 编码 ， 取 而 代 之 的 是 从 文件 中 读 取 需 要 的 事件 (第 12 章 的 练习 会 要 求 读者 照 此 方法 
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修改 这 个 例子 )。 

这 个 例子 应 该 使 读者 更 了 解 内 部 类 的 价值 了 ， 特 别 是 在 控制 框架 中 使 用 内 部 类 的 时 候 。 在 
第 18 章 中 ， 读 者 将 看 到 内 部 类 如 何 优雅 地 描述 图 形 用 户 界面 的 行为 。 到 那 时 ， 读 者 应 该 就 完全 
信服 内 部 类 的 价值 了 。 

练习 24: (2) 在 GreenhouseControlsjava 中 增加 一 个 Event 内 部 类 ， 用 以 打开 、 关 闭 风扇 。 
配置 GreenhouseControlierjaval 以 使 用 这 些 新 的 Event 对 象 。 

练习 25: (3) 在 GreenhouseControls.java 中 继承 GreenhouseControls， 增 加 Event 内 部 类 ， 用 
以 开启 、 关 闭 喷 水 机 。 写 一 个 新 版 的 GreenhouseControllerjava 以 使 用 这 些 新 的 Event 对 象 。 


10.9 内 部 类 的 继承 


因为 内 部 类 的 构造 器 必须 连接 到 指向 其 外 围 类 对 象 的 引用 ， 所 以 在 继承 内 部 类 的 时 候 ， 事 
情 会 变 得 有 点 复杂 。 问 题 在 于 ， 那 个 指向 外 围 类 对 象 的 “秘密 的 ”引用 必须 被 初始 化 ， 而 在 导 
出 类 中 不 再 存在 可 连接 的 默认 对 象 。 要 解决 这 个 问题 ， 必 须 使 用 特殊 的 语法 来 明确 说 清 它们 之 
间 的 关联 ; 

71/: innerclasses/InheritInner .java 


// Inheriting an inner class. 


class WithInner { 
class Inner {} 


} 


public class InheritInner extends WithInner.Inner { 
//! InheritInner() {} // Won't compile 
InheritInner(WithInner wi) { 
wi. super (); 


} 

public static void mafn(String[] args) { 
WithInner wi = new WithInner(); 
InheritInner ii = new InheritInner (wi): 


} 
} Mi~ 


可 以 看 到 ，InheritInner 只 继承 自 内 部 类 ， 而 不 是 外 围 类 。 但 是 当 要 生成 一 个 构造 器 时 ， 默 
认 的 构造 器 并 不 算 好 ， 而 且 不 能 只 是 传递 一 个 指向 外 围 类 对 象 的 引用 。 此 外 ， 必 须 在 构造 器 内 
使 用 如 下 语法 : 

enclos ingClassReference. super () : 
这 样 才 提 供 了 必要 的 引用 ， 然 后 程序 才能 编译 通过 。 

练习 26: (2) 创建 一 个 包含 内 部 类 的 类 ， 此 内 部 类 有 一 个 非 默认 的 构造 器 (需要 参数 的 构造 
器 )。 创 建 另 一 个 也 包含 内 部 类 的 类 ， 此 内 部 类 继承 自 第 一 个 内 部 类 。 


10.10 内 部 类 可 以 被 覆盖 吗 


如 果 创 建 了 一 个 内 部 类 ， 然 后 继承 其 外 围 类 并 重新 定义 此 内 部 类 时 ， 会 发 生 什么 呢 ? 也 就 
是 说 ， 内 部 类 可 以 被 覆盖 吗 ? 这 看 起 来 似乎 是 个 很 有 用 的 思想 ， 但 是 “覆盖 ”内 部 类 就 好 像 它 
是 外 围 类 的 一 个 方法 ， 其 实 并 不 起 什么 作用 : 


//: innerclasses/Bigegg.java 
// An inner class cannot’ be overriden like a method. 
import static net.mindview.util.Print.*; 
class Egg { 
private Yolk y: 
protected class Yolk { 
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public Yolk() { print(“Egg.Yolk()"); } 


) 
public Egg() { 
print ("New Egg()"); 
y = new Youk(); 
} 
$ 


public class BigEgg extends Egg { 
public class Yolk { 
public Yolk() { print("BigEgg.Yolk()"); } 


} 
public static void main(String[] args) { 
new Bigtgg(); 


} 
} /* Output: 
New Egg() 
Egg. Yolk() 
“Whim 


默认 的 构造 器 是 编译 器 自动 生成 的 ， 这 里 是 调用 基 类 的 默认 构造 器 。 你 可 能 认为 既然 创建 
了 BigEgg 的 对 象 ， 那 么 所 使 用 的 应 该 是 “覆盖 后 ”的 Yolk 版 本 ， 但 从 输出 中 可 以 看 到 实际 情况 
并 不 是 这 样 的 。 

这 个 例子 说 明 ， 当 继承 了 某 个 外 围 类 的 时 候 ， 内 部 类 并 没有 发 生 什么 特别 神奇 的 变化 。 这 
两 个 内 部 类 是 完全 独立 的 两 个 实体 ， 各 自在 自己 的 命名 空间 内 。 当 然 ， 明 确 地 继承 某 个 内 部 类 
也 是 可 以 的 ， 


//: Annerclasses/Bigegg2. java 
// Proper inheritance of an inner class. 
import static net.mindview.util.Print.*; 


class Egg? { 
protected class Yolk { 
public Yolk() { print("Egg2.Yolk()"); } 
public void f() { print("Egg2.Yolk.f()");:} 
} 
private Yolk y = new Yolk(); 
public Egg2() { print("New Egg2()"): } 
public void insertYolk(Yolk yy) { y = yy: } 
public void g0 { y.f(; } 
} 


public class BigEgg2 extends Egg? ( 
public class Yolk extends Egg2.Yolk { 
public Yolk() { print("Bigégg2.Yolk()"); } 
public void f() { print("Bigégg2.Yolk.f()"); } 


$ 
public BigEgg2() { insertYolk(new Yolk()): } 
public static void main(String[] args) { 
Egg2 e2 = new BigEgg2(); 
€2.80; 
} 
} /* Output: 
Egg2.Yolk() 
New Egg2() 
Eg82. Yolk() 
BigEgg2. Yolk() 
BigEgg2. Yolk.f() 
“Wha 


现在 BigEgg2.Yolk 通 过 extends Egg2.Yolk 明 确 地 继承 了 此 内 部 类 ， 并 且 覆 盖 了 其 中 的 方法 。 
insertYolk0 方 法 允许 BigEgg2 将 它 自己 的 Yolk 对 象 向 上 转型 为 Egg2 中 的 引用 y。 所 以 当 g0 调 用 
yf0 时 ， 覆 盖 后 的 新 版 的 f0 被 执行 。 第 二 次 调用 Egg2.Yolk0， 结 果 是 BigEgg2.Yolk 的 构造 器 调用 
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了 其 基 类 的 构造 器 。 可 以 看 到 在 调用 g0 的 时 候 ， 新 版 的 f0 被 调用 了 。 
10.11 局 部 内 部 类 


前 面 提 到 过 ， 可 以 在 代码 块 里 创建 内 部 类 ， 典 型 的 方式 是 在 一 个 方法 体 的 里 面 创建 。 局 部 
内 部 类 不 能 有 访问 说 明 符 ， 因 为 它 不 是 外 围 类 的 一 部 分 ， 但 是 它 可 以 访问 当前 代码 块 内 的 常量 ， 
以 及 此 外 围 类 的 所 有 成 员 。 下 面 的 例子 对 局 部 内 部 类 与 匿名 内 部 类 的 创建 进行 了 比较 。 


//: innerclasses/LocalInnerClass. java 
// Holds a sequence of Objects. 
import static net.mindview.util.Print.*; 


interface Counter { 
int next(); 


public class LocalInnerClass { 
private int count = @; 
Counter getCounter(final String name) { 
// A local inner class: 
class LocalCounter implements Counter { 
public LocalCounter() { 
// Local inner class can have a constructor 
print("LocalCounter()"); 
} 
public int next() { 
printnb(name); // Access local final 
return count++; 


$ 


return new LocalCounter(); 
} 
// The same thing with an anonymous inner class: 
Counter getCounter2(final String name) { 
return new Counter() { 
// Anonymous inner class cannot have a named 
// constructor, only an instance initializer: 


print("Counter()"); 


public int next() { 
printnb(name); // Access local final 
return count++; 
} 
k 
£ 
public static void main(String[] args) { 
LocalInnerClass lic = new LocalInnerClass() ; 
Counter 
cl = lic.getCounter ("Local inner "), 
c2 = lic.getCounter2 ("Anonymous inner "); 
for(int 1 = 6; 1 <5; i++) 
print(cl.next()); 
for(int i = 0; i < 5; 1++) 
print(c2.next()); 





} 

} /* Output: 
LocalCounter() 
Counter () 

Local inner 
Lotal inner 
Local inner 
Local inner 
Local inner 


awuNre 
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Anonymous inner 5 
Anonymous inner 6 
Anonymous inner 7 
Anonymous inner 8 
Anonymous inner 9 
UW 


Counter 返 回 的 是 序列 中 的 下 一 个 值 。 我 们 分 别 使 用 局 部 内 部 类 和 匿名 内 部 类 实现 了 这 个 功 
能 ， 它 们 具有 相同 的 行为 和 能 力 。 既 然 局 部 内 部 类 的 名 字 在 方法 外 是 不 可 见 的 ， 那 为 什么 我 们 
仍然 使 用 局 部 内 部 类 而 不 是 匿名 内 部 类 呢 ? 唯一 的 理由 是 ， 我 们 需要 一 个 已 命名 的 构造 器 ， 或 
者 需要 重 载 构造 器 ， 而 匿名 内 部 类 只 能 用 于 实例 初始 化 。 

所 以 使 用 局 部 内 部 类 而 不 使 用 匿名 内 部 类 的 另 一 个 理由 就 是 , 需要 不 止 一 个 该 内 部 类 的 对 象 。 


10.12 内 部 类 标识 符 


由 于 每 个 类 都 会 产生 一 个 .class 文 件 ， 其 中 包含 了 如 何 创建 该 类 型 的 对 象 的 全 部 信息 (此 信 

息 产 生 一 个 “meta-class”， 岂 做 Class 对 象 )， 你 可 能 猜 到 了 ， 内 部 类 也 必须 生成 一 个 .class 文 件 以 

， 包 含 它们 的 Class 对 象 信息 。 这 些 类 文件 的 命名 有 严格 的 规则 ， 外 围 类 的 名 字 ， 加 上 “$”， 再 加 
上 内 部 类 的 名 字 。 例 如 ，LocalInnerClassjava 生 成 的 .class 文 件 包括 : 


Counter .class 
LocalInnerClass$1.class 
LocalInnerClass$1LocalCounter.class 
LocalInnerClass.class 


如 果 内 部 类 是 匿名 的 ， 编 译 器 会 简单 地 产生 一 个 数字 作为 其 标识 符 。 如 果 内 部 类 是 侈 套 在 
别 的 内 部 类 之 中 ， 只 需 直接 将 它们 的 名 字 加 在 其 外 围 类 标识 符 与 “$” 的 后 面 。 

虽然 这 种 命名 格式 简单 而 直接 ， 但 它 还 是 很 健壮 的 ， 足 以 应 对 绝 大 多 数 情况 。 因 为 这 是 
Java 的 标准 命名 方式 ， 所 以 产生 的 文件 自动 都 是 平台 无 关 的 。( 注 意 ， 为 了 保证 你 的 内 部 类 能 起 
作用 ，Java 编 译 器 会 尽 可 能 地 转换 它们 。) 


10.13 总 结 


比 起 面向 对 象 编程 中 其 他 的 概念 来 ， 接 口 和 内 部 类 更 深奥 复杂 ， 比 如 C++ 就 没有 这 些 。 将 
两 者 结合 起 来 ， 同 样 能 够 解决 C++ 中 的 用 多 重 继承 所 能 解决 的 问题 。 然 而 ， 多 重 继承 在 C++ 中 被 
证 明 是 相当 难以 使 用 的 ， 相 比较 而 言 ，Java 的 接口 和 内 部 类 就 容易 理解 多 了 。 
虽然 这 些 特性 本 身 是 相当 直观 的 ， 但 是 就 像 多 态 机 制 一 样 ， 这 些 特性 的 使 用 应 该 是 设计 阶 
段 考虑 的 问题 。 随 着 时 间 的 推移 ， 读 者 将 能 够 更 好 地 识别 什么 情况 下 应 该 使 用 接口 ， 什 么 情况 
使 用 内 部 类 ， 或 者 两 者 同时 使 用 。 但 此 时 ， 读 者 至 少 应 该 已 经 完全 理解 了 它们 的 语法 和 语义 。 
当 见 到 这 些 语言 特性 实际 应 用 时 ， 就 最 终 理 解 它们 了 。 
所 选 习题 的 答案 都 可 以 在 名 为 The Thinking in Java Annotated Solution Guide 的 电子 文档 中 找 
到 ， 读 者 可 以 从 www.MindView net 购 买 此 文档 。 388 














日 而 务 一 方面 ， 对 于 Unix shell 而 言 ，“$” 是 一 个 元 字符 ， 所 以 在 列 出 .class 文件 的 时 候 ， 有 时 会 有 问题 。 这 对 于 
基于 Unix 的 Sun 公 司 而 言 ， 真 是 有 点 奇怪 。 我 猿 这 是 因为 他 们 没有 考虑 这 个 问题 ， 他 们 认为 你 自然 是 应 该 专注 
于 源码 文件 的 。 





第 11 章 持 有 对 象 


如 果 一 个 程序 只 包含 固定 数量 的 且 其 生命 期 都 是 已 知 的 对 象 ， 那 么 这 是 一 个 非常 简单 的 
程序 。 
通常 ， 程 序 总 是 根据 运行 时 才 知 道 的 某 些 条 件 去 创建 新 对 象 。 在 此 之 前 ， 不 会 知道 所 需 对 
象 的 数量 ， 甚 至 不 知道 确切 的 类 型 。 为 解决 这 个 普遍 的 编程 问题 ， 需 要 在 任意 时 刻 和 任意 位 置 
创建 任意 数量 的 对 象 。 所 以 ， 就 不 能 依靠 创建 命名 的 引用 来 持 有 每 一 个 对 象 : 

MyType aReference; 

1 ”因为 你 不 知道 实际 上 会 需要 多 少 这 样 的 引用 。 

大 多 数 语言 都 提供 某 种 方法 来 解决 这 个 基本 问题 。Java 有 多 种 方式 保存 对 象 (应 该 说 是 对 
象 的 引用 )。 例 如 前 面 曾经 学 习 过 的 数组 ， 它 是 编译 器 支持 的 类 型 。 数 组 是 保存 一 组 对 象 的 最 有 
效 的 方式 ， 如 果 你 想 保存 一 组 基本 类 型 数据 ， 也 推荐 使 用 这 种 方式 。 但 是 数组 具有 固定 的 尺寸 
而 在 更 一 般 的 情况 中 ， 你 在 写 程序 时 并 不 知道 将 需要 多 少 个 对 象 ， 或 者 是 否 需要 更 复杂 的 方式 
来 存储 对 象 ， 因 此 数组 尺寸 固定 这 一 限制 显得 过 于 受 限 了 。 

Java 实用 类 库 还 提供 了 一 套 相 当 完 整 的 容器 类 来 解决 这 个 问题 ， 其 中 基本 的 类 型 是 List、 
Set、Queue 和 Map。 这 些 对 象 类 型 也 称 为 集合 类 ， 但 由 于 Java 的 类 库 中 使 用 了 Collection 这 个 名 
字 来 指 代 该 类 库 的 一 个 特殊 子 集 ， 所 以 我 使 用 了 范围 更 广 的 术语 “容器 ”称呼 它们 。 容 器 提供 
了 完善 的 方法 来 保存 对 象 ， 你 可 以 使 用 这 些 工具 来 解决 数量 惊人 的 问题 。 

容器 还 有 其 他 一 些 特性 。 例 如 ，Set 对 于 每 个 值 都 只 保存 一 个 对 象 ，Map 是 允许 你 将 某 些 对 
象 与 其 他 一 些 对 象 关联 起 来 的 关联 数组 ，Java 容 器 类 都 可 以 自动 地 调整 自己 的 尺寸 。 i4 
数组 不 同 ， 在 编程 时 ， 你 可 以 将 任意 数量 的 对 象 放 置 到 容器 中 ， 并 且 不 需要 担心 容器 应 该 设置 
为 多 大 。 

即使 在 Java 中 没有 直接 的 关键 字 支 持 * ， 容 器 类 仍旧 是 可 以 显著 增强 你 的 编程 能 力 的 基本 工 
具 。 在 本 章 中 ， 你 将 了 解 有 关 jJava 容 器 类 库 的 基本 知识 ， 以 及 对 典型 用 法 的 重点 介绍 。 我 们 聚 
焦 于 你 在 日 复 一 日 的 编程 工作 中 将 会 用 到 的 那些 容器 。 稍 后 ， 在 第 17 章 ， 还 将 学 习 到 剩余 的 那 
些 容器 ， 以 及 有 关 它 们 的 功能 和 如 何 使 用 它们 的 更 多 细节 。 


11.1 泛 型 和 类 型 安全 的 容器 


使 用 Java SE5 之 前 的 容器 的 一 个 主要 问题 就 是 编译 器 允许 你 向 容器 中 插入 不 正确 的 类 型 。 例 
如 ， 考 虑 一 个 Apple 对 象 的 容器 ， 我 们 使 用 最 基本 最 可 靠 的 容器 ArrayList。 现 在 ， 你 可 以 把 
ArrayList 当 作 “ 可 以 自动 扩充 自身 尺寸 的 数组 ”来 看 待 。 使 用 ArrayList 相 当 简单 ;创建 一 个 实 
例 ， 用 add0 插 入 对 象 ， 然 后 用 get0 访 问 这 些 对 象 ， 此 时 需要 使 用 索引 ， 就 像 数组 一 样 ， 但 是 不 
需要 方 括号 ” 。ArrayList 还 有 一 个 size0 方 法 ， 使 你 可 以 知道 已 经 有 多 少 元 素 添加 了 进来 ， 从 而 
不 会 不 小 心 因素 引 越界 而 引发 错误 〈 通 过 抛 出 运行 时 异常 ， 异 常 将 在 第 12 章 介绍 )。 





在 本 例 中 ，Apple 和 Orange 都 放置 在 了 容器 中 ， 然 后 将 它们 取出 。 正 常情 况 下 ，Java 编 译 器 


O 许多 语言 ,例如 Perl、Python 和 Ruby， 都 有 容器 的 本 地 支持 。 
O 这 里 是 操作 符 重 载 的 用 武之 地 ，C++ 和 C# 的 容器 类 都 使 用 操作 符 重 载 生成 了 更 简洁 的 语法 。 
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会 报告 警告 信息 ， 因 为 这 个 示例 没有 使 用 泛 型 。 在 这 里 ， 我 们 使 用 Java SE5 所 特有 的 注解 来 抑制 


了 警告 信息 。 注 解 以 “@” 符 号 开头 ， 可 以 接受 参数 ， 这 里 的 @SuppressWarnings 注 解 及 其 参数 ” [399] 


表示 只 有 有 关 “ 不 受 检查 的 异常 ”的 警告 信息 应 该 被 抑制 : 


//: holding/ApplesAndOr angesWi thoutGenerics. java 

// Simple container example (produces compiler warnings). 
// {ThrowsException} 

import java.util.*; 


class Apple { 
private static long counter; 
private final long id = counter++; 
public long id() { return id; } 


class Orange {} 


public class ApplesAndOrangeswithoutGenerics { 
@SuppressWarnings ("unchecked") 
public static void main(String[] args) { 
ArrayList apples = new ArrayList(): 
for(int 1 = @; i < 3; 1++) 
apples.add(new Apple()); 
// Not prevented from adding an Orange to apples: 
apples.add(new Orange()) : 
for(int 1 = 0; f < apples.size(); i++) 
(Apple) apples.get(1)).1d(); 
// Orange is detected only at run time 


} 
} /* (Execute to see output) *///:~ 


在 第 20 章 中 将 会 更 多 地 学 习 有 关 Java SE5 的 注解 。 

Apple 和 Orange 类 是 有 区 别 的 ， 它 们 除了 都 是 Object 之 外 没有 任何 共性 ( 记 住 ， 如 果 一 个 类 
没有 显 式 地 声明 继承 自 哪个 类 ， 那 么 它 自动 地 继承 自 Object) 。 因 为 ArrayList 保 存 的 是 Object， 
因此 你 不 仅 可 以 通过 ArrayList 的 add0 方 法 将 Apple 对 象 放 进 这 个 容器 ， 还 可 以 添加 Orange 对 象 ， 
而 且 无 论 在 编译 期 还 是 运行 时 都 不 会 有 问题 。 当 你 在 使 用 ArrayList 的 getO 方 法 来 取出 你 认为 是 
Apple 的 对 象 时 ， 你 得 到 的 只 是 Object 引用 ， 必 须 将 其 转型 为 Apple， 因 此 ， 需 要 将 整个 表达 式 
括 起 来 ， 在 调用 Apple 的 id( 方 法 之 前 ， 强 制 执行 转型 。 否 则 ， 你 就 会 得 到 语法 错误 。 在 运行 时 ， 
当 你 试图 将 Orange 对 象 转型 为 Apple 时 ， 你 就 会 以 前 面 提 及 的 异常 的 形式 得 到 一 个 错误 。 

在 第 15 章 中 ， 你 将 会 了 解 到 ， 使 用 Java 泛 型 来 创建 类 会 非常 复杂 。 但 是 ， 应 用 预定 义 的 泛 
型 通常 会 很 简单 。 例 如 ， 要 想 定义 用 来 保存 Apple 对 象 的 ArrayList， 你 可 以 声明 
ArrayList<Apple>， 而 不 仅仅 只 是 ArrayList， 其 中 央 括 号 括 起 来 的 是 类 型 参数 (可 以 有 多 个 )， 
它 指定 了 这 个 容器 实例 可 以 保存 的 类 型 。 通 过 使 用 泛 型 ， 就 可 以 在 编译 期 防止 将 错误 类 型 的 对 
象 放置 到 容器 中 。。 下 面 还 是 这 个 示例 ， 但 是 使 用 了 泛 型 : 

//: holding/ApplesAndOrangesWithGenerics. java 

import java.util.*; 


public class ApplesAndOrangesWithGenerics { 
public static void main(String{] args) { 
ArrayList<Apple> apples = new ArrayList<Apple>(): 
for(int 1 = 0; 1 < 3; i++) 
apples. add(new Apple()); 
// Compile-time error: 


日 在 第 15 章 的 未 尾 ， 你 会 发 现 有 关 这 个 问题 是 否 很 严重 的 讨论 。 但 是 ， 第 15 章 还 将 展示 Java 泛 型 远 不 止 类 型 安全 
的 容器 这 么 简单 。 
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// apples.add(new Orange()); 
for(int i = @; i < apples.size(); i++) 
System. out.printIn(apples.get(i).id()); 
41 Using foreach: 
for (Apple c : apples) 
System.out. printin(c.1d()); 


} 
/* Output: 


现在 ， 编 译 器 可 以 阻止 你 将 Orange 放 置 到 apples 中 ， 因 此 它 变 成 了 一 个 编译 期 错误 ， 而 不 
再 是 运行 时 错误 。 

你 还 应 该 注意 到 ， 在 将 元 素 从 List 中 取出 时 ， 类 型 转换 也 不 再 是 必需 的 了 。 因 为 List 知 道 它 
保存 的 是 什么 类 型 ， 因 此 它 会 在 调用 getO 时 替 你 执行 转型 。 这 样 ， 通 过 使 用 泛 型 ， 你 不 仅 知道 
编译 器 将 会 检查 你 放置 到 容器 中 的 对 象 类 型 ， 而 且 在 使 用 容器 中 的 对 象 时 ， 可 以 使 用 更 加 清晰 
的 语法 。 

这 个 实例 还 表明 ， 如 果 不 需 要 使 用 每 个 元 素 的 索引 ， 你 可 以 使 用 foreach 语 法 来 选择 List 中 的 
每 个 元 素 。 Z 

当 你 指定 了 某 个 类 型 作为 泛 型 参数 时 ， 你 并 不 仅 限于 只 能 将 该 确切 类 型 的 对 象 放置 到 容器 
中 。 向 上 转型 也 可 以 像 作用 于 其 他 类 型 一 样 作用 于 泛 型 : 


//: holding/GenericsAndUpcasting. java 
import java.util.*; 


class GrannySmith extends Apple () 
class Gala extends Apple {} 

class Fuji extends Apple {} 

class Braeburn extends Apple {} 


public class GenericsAndUpcasting { 
public static void main(String) args) { 
ArrayList<Apple> apples = new ArrayList<Apple>(); 
apples. add(new GrannySmith()) ; 
apples.add(new Gala()); 
apples.add(new Fuji()): 
apples. add(new Braeburn()); 
for (Apple c : apples) 
System.out.printin(c); 


} te Output: (Sample) 

GrannySmith@7d772e 

Gala@11b86e7 

Fuji@35ce36 

Braeburn@757aef 

#77:~ 

因此 ， 你 可 以 将 Apple 的 子 类 型 添加 到 被 指定 为 保存 Apple 对 象 的 容器 中 。 

程序 的 输出 是 从 Object 默 认 的 toString0 方 法 产生 的 ， 该 方法 将 打印 类 名 ， 后 面 跟随 该 对 象 
的 散 列 码 的 无 符号 十 六 进 制 表示 (这 个 散 列 码 是 通过 hashCode0 方 法 产生 的 )。 你 将 在 第 17 章 中 
了 解 有 关 散 列 码 的 内 容 。 

练习 1: (2) 创建 一 个 新 类 Gerbil ( 沙 鼠 ) ， 包 含 int gerbilNumber， 在 构造 器 中 初始 化 它 


《类似 于 本 章 的 Mouse 示 例 )。 添 加 一 个 方法 hop0, 用 以 打印 沙 鼠 的 号 码 以 及 它 正 在 跳跃 的 信息 。 
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创建 一 个 ArrayList， 并 向 其 中 添加 一 串 Gerbil 对 象 。 使 用 getO 遍 历 List， 并 且 对 每 个 Gerbil 调 
用 hop0。 


11.2 基本 概念 


Java 容器 类 类 库 的 用 途 是 “保存 对 象 "， 并 将 其 划分 为 两 个 不 同 的 概念 

1) Collection。 一 个 独立 元 素 的 序列 ， 这 些 元 素 都 服从 一 条 或 多 条 规则 。List 必须 按照 插入 
的 顺序 保存 元 素 ， 而 Set 不 能 有 重复 元 素 。Queue 按 照排 队 规则 来 确定 对 象 产生 的 顺序 (通常 与 它 
们 被 插入 的 顺序 相同 )。 

2) Map。 一 组 成 对 的 “ 键 值 对 ”对 象 ， 人 允许 你 使 用 键 来 查找 值 。ArrayList 允 许 你 使 用 数字 
来 查找 值 ， 因 此 在 某 种 意义 上 讲 ， 它 将 数字 与 对 象 关 联 在 了 一 起 。 瑞 射 表 允许 我 们 使 用 另 一 个 
对 象 来 查找 某 个 对 象 ， 它 也 被 称 为 “关联 数组 "， 因 为 它 将 某 些 对 象 与 另外 一 些 对 象 关 联 在 了 一 
起 ， 或 者 被 称 为 “字典 "， 因 为 你 可 以 使 用 键 对 象 来 查找 值 对 象 ， 就 像 在 字典 中 使 用 单词 来 定义 
一 样 。Map 是 强大 的 编程 工具 。 

尽管 并 非 总 是 这 样 ， 但 是 在 理想 情况 下 ， 你 编写 的 大 部 分 代码 都 是 在 与 这 些 接口 打交道 ， 
并 且 你 唯一 需要 指定 所 使 用 的 精确 类 型 的 地 方 就 是 在 创建 的 时 候 。 因 此 ， 你 可 以 像 下 面 这 样 创 
建 一 个 List， 

List<Apple> apples = new ArrayList<Apple>(); 

注意 ，ArrayList 已 经 被 向 上 转型 为 List， 这 与 前 一 个 示例 中 的 处 理 方式 正好 相反 。 使 用 接 
口 的 目的 在 于 如 果 你 决定 去 修改 你 的 实现 ， 你 所 需 的 只 是 在 创建 出 修改 它 ， 就 像 下 面 这 样 : 

List<Apple> apples = new LinkedList<Apple>(); 

因此 ， 你 应 该 创建 一 个 具体 类 的 对 象 ， 将 其 转型 为 对 应 的 接口 ， 然 后 在 其 余 的 代码 中 都 使 
用 这 个 接口 。 

这 种 方式 并 非 总 能 奏效 ， 因 为 某 些 类 具有 额外 的 功能 ， 例 如 ，LinkedList 具 有 在 List 接 口中 
未 包含 的 额外 方法 ， 而 TreeMap 也 具有 在 Map 接 口中 未 包含 的 方法 。 如 果 你 需要 使 用 这 些 方法 ， 
就 不 能 将 它们 向 上 转型 为 更 通用 的 接口 。 

Collection 接 口 概括 了 序列 的 概念 一 一 一 种 存放 一 组 对 象 的 方式 。 下 面 这 个 简单 的 示例 用 
Integer 对 象 填充 了 一 个 Collection (这 里 用 ArrayList 表 示 ) ， 然 后 打印 所 产生 的 容器 中 的 所 有 
TH: 

//: holding/SimpleCollection. java 

import java.util.*; 


public class SimpleCollection { 
public static void main(String[] args) { 
Collection<Integer> c = new ArrayList<Integer>(); 
for(int i = 6; i < 10; i++) 
c.add(i); // Autoboxing 
for (Integer i : c) 
System.out.print(i +", "); 


} 
} /* Output: 


6, 1, 2,3, 4, 5, 6, 7, 8 9 
“H~ 


因为 这 个 示例 只 使 用 了 Collection 方 法 ， 因 此 任何 继承 自 Collection 的 类 的 对 象 都 可 以 正常 工 
fe, 但 是 ArrayList 是 最 基本 的 序列 类 型 。 

add0 方 法 的 名 称 就 表明 它 是 要 将 一 个 新 元 素 放置 到 Collection 中 。 但 是 ， 文 档 中 非常 仔细 地 
叙述 到 :“ 要 确保 这 个 Collection 包 含 指定 的 元 素 .” 这 是 因为 考虑 到 了 Set 的 含义 ， 因 为 在 Set 中 
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只 有 元 素 不 存在 的 情况 下 才 会 添加 。 在 使 用 ArrayList， 或 者 任何 种 类 的 List 时 ，add0 总 是 表示 
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“把 它 放 进 去 ”， 因 为 List 不 关心 是 否 存在 重复 。 

所 有 的 Collection 都 可 以 用 foreach 语 法 遍历 ， 就 像 这 里 所 展示 的 。 在 本 章 的 后 续 部 分 ， 你 将 
会 学 习 到 被 称 为 “过 代 器 ”的 更 灵活 的 概念 。 

练习 2: (1) 修改 SimpleCollectionjava， 使 用 Set 来 表示 c。 

练习 3: (2) 修改 innerclasses/Sequence.java， 使 你 可 以 向 其 中 添加 任意 数量 的 元 素 。 


11.3 添加 一 组 元 素 


在 javautil 包 中 的 Arrays 和 Collections 类 中 都 有 很 多 实用 方法 ， 可 以 在 一 个 Collection 中 添加 
一 组 元 素 。Arrays.asList0 方 法 接受 一 个 数组 或 是 一 个 用 逗号 分 隔 的 元 素 列表 (使 用 可 变 参数 ) ， 
并 将 其 转换 为 一 个 List 对 象 。Collections.addAll0 方 法 接受 一 个 Collection 对 象 ， 以 及 一 个 数组 或 
是 一 个 用 逗号 分 割 的 列表 ， 将 元 素 添加 到 Collection 中 。 下 面 的 示例 展示 了 这 两 个 方法 ， 以 及 更 
加 传统 addAll0 方 法 ， 所 有 Collection 类 型 都 包含 该 方法 : 


//: holding/AddingGroups. java 
// Adding groups of elements to Collection objects. 
import java.util.*; 


public class AddingGroups { 
public static void main(String[] args) { 
Collection<Integer> collection = 
new ArrayList<Integer>(Arrays.asList(1, 2, 3, 4, 5)): 
Integer[] moreInts = { 6, 7, 8, 9, 18 }; 
collection. addAll (Arrays.asList(moreInts)); 
// Runs significantly faster, but you can't 
// construct a Collection this way: 
Collections.addAll(collection, 11, 12, 13, 14, 15); 
Collections. addAll (collection, moreInts); 
// Produces a list “backed by" an array: 
List<Integer> list = Arrays.asList(16, 17, 18, 19, 20); 
List.set(1, 99); // OK -- modify an element 
// Vist.add(21); // Runtime error because the 
// underlying array cannot be resized, 


} 

} i~ 

Collection 的 构造 器 可 以 接受 另 一 个 Collection， 用 它 来 将 自身 初始 化 ， 因 此 你 可 以 使 用 
Arrays.List0 来 为 这 个 构造 器 产生 输入 。 但 是 ，Collection.addAli0 方 法 运行 起 来 要 快 得 多 ,而 且 
构建 一 个 不 包含 元 素 的 Colleetion， 然 后 调用 Collections.addAHO 这 种 方式 很 方便 ， 因 此 它 是 首 
选 方式 。 

Collection.addAll() 成 员 方 法 只 能 接受 另 一 个 Collection 对 象 作为 参数 ， 因 此 它 不 如 
Arrays.asList0 或 Collections.addAli0 灵 活 ， 这 两 个 方法 使 用 的 都 是 可 变 参 数列 表 。 

你 也 可 以 直接 使 用 Arrays.asList0 的 输出 ， 将 其 当 作 List， 但 是 在 这 种 情况 下 ， 其 底层 表示 
的 是 数组 ， 因 此 不 能 调整 尺寸 。 如 果 你 试图 用 add0 或 delete0 方 法 在 这 种 列表 中 添加 或 删除 元 素 ， 
就 有 可 能 会 引发 去 改变 数组 尺寸 的 尝试 ， 因 此 你 将 在 运行 时 获得 “Unsupported Operation (不 支 
持 的 操作 )” 错 误 。 

Arrays.asList0 方 法 的 限制 是 它 对 所 产生 的 List 的 类 型 做 出 了 最 理想 的 假设 ， 而 并 没有 注意 
你 对 它 会 赋予 什么 样 的 类 型 。 有 时 这 就 会 引发 问题 : 


//: holding/AsListInference. java 
// Arrays.asList() makes its best guess about type. 
import java.util.*: 
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class Snow {} 

class Powder extends Snow {} 
class Light extends Powder {} 
class Heavy extends Powder {} 
class Crusty extends Snow {} 
class Slush extends, Snow {} 


public class AsListInference { 
public static void main(String[] args) { 
List<Snow> snowl = Arrays.asList( 
new Crusty(), new Slush(), new Powder()); 


// Won't compile: 

1/ List<Snow> snow2 = Arrays.asList( 

// new Light(), new Heavy()); 

// Compiler says: 

// found : java.util.List<Ponder> 

// required: java.util.List<Snow> 

// Collections. addAll() doesn't get confused: 
List<Snow> snow3 = new ArrayList<Snow>(); 
Collections.addAl1(snow3, new Light(), new Heavy()); 





// Give a hint using an 

// explicit type argument specification: 

List<Snow> snow4 = Arrays.<Snow>asList( 
new Light(), new Heavy()); 


} 有， 

} Mi~ 

当 试图 创建 saow2 时 ，Arrays.asList0 中 只 有 Powder 类 型 ， 因 此 它 会 创建 List<Powder> 而 不 
是 List<Snow>， 尽 管 Collections.addAlI0 工 作 的 很 好 ， 因 为 它 从 第 一 个 参数 中 了 解 到 了 目标 类 型 
是 什么 。 

正如 你 从 创建 snow4 的 操作 中 所 看 到 的 ， 可 以 在 Arrays.asList0 中 间 插 入 一 条 “线索 ”， 以 告 
诉 编译 器 对 于 由 Arrays.asList0 产 生 的 List 类 型 ， 实 际 的 目标 类 型 应 该 是 什么 。 这 称 为 显 式 类 型 
参数 说 明 。 

正如 你 所 见 ，Map 更 加 复杂 ， 并 且 除 了 用 另 一 个 Map 之 外 ，Java 标 准 类 库 没有 提供 其 他 任 
何 自动 初始 化 它们 的 方式 。 


11.4 容器 的 打印 


你 必须 使 用 Arrays.toString0 来 产生 数组 的 可 打印 表示 ， 但 是 打印 容器 无 需 任 何 帮 助 。 下 面 
是 一 个 例子 ， 这 个 例子 中 也 介绍 了 一 些 基本 类 型 的 容器 : 


/1: holding/PrintingContainers. java 

// Containers print themselves automatically. 
import java.util.*; 

import static net.mindview.util.Print.*; 


public class PrintingContainers { 

static Collection fill (Collection<String> collection) { 
collection. add("rat" 
collection. add ("cat 
collection. add{"dog") 
collect ion. add("dog") ; 
return collection; 

} 398] 

static Map fill(Map<String,String> map) { 
map.put("rat", "Fuzzy"); 
map.put(“cat", "Rags"); 
map.put("dog", "Bosco"); 















222 RUE 





map.put("dog", “Spot"): 
return map; 


} 

public static void main(String{] args) { 
print(fill(new ArrayList<String>())): 
print(fill(new LinkedList<String>())); 
print(fill(new HashSet<String>())); 
print(fitl(new TreeSet<String>())); 
print(fill(new LinkedHashSet<String>())); 
print(fill(new HashMap<String, String>())); 
print(fill(new TreeMap<String, String>())); 
print(fill(new LinkedHashMap<String, String>())); 


} 六 Output: 

[rat, cat, dog, dog) 

[rat, cat, dog, dog) 

(dog, cat, rat] 

(cat, dog, rat] 

(rat, cat, dog) 

{dog=Spot, cat=Rags, rat=Fuzzy} 

{cat=Rags, dog=Spot, rat=Fuzzy)} 

{rat=Fuzzy, cat=Rags, dog=Spot} 

Whi~ 

这 里 展示 了 Java 容 器 类 库 中 的 两 种 主要 类 型 ， 它 们 的 区 别 在 于 容器 中 每 个 “ 横 ” 保 存 的 元 
素 个 数 。Collection 在 每 个 模 中 只 能 保存 一 个 元 素 。 此 类 容器 包括 : List， 它 以 特定 的 顺序 保存 
一 组 元 素 ，Set， 元 素 不 能 重复 ，Queue， 只 人 允许 在 容器 的 一 “ 端 ” 插 入 对 象 ， 并 从 另外 一 “ 端 ” 
移 除 对 象 (对 于 本 例 来 说 ， 这 只 是 另外 一 种 观察 序列 的 方式 ， 因 此 并 没有 展示 它 )。Map 在 每 个 
档 内 保存 了 两 个 对 象 ， 即 键 和 与 之 相关 联 的 值 。 

查看 输出 会 发 现 ， 默 认 的 打印 行为 〈 使 用 容器 提供 的 toString0 方 法 ) 即 可 生成 可 读 性 很 好 
的 结果 。Colleetion 打 印 出 来 的 内 容 用 方 括号 括 住 ， 每 个 元 素 由 逗号 分 隔 。Map 则 用 大 括号 括 住 ， 
键 与 值 由 等 号 联系 〈 键 在 等 号 左边 ， 值 在 右边 ) 。 

第 一 个 f10 方 法 可 以 作用 于 所 有 类 型 的 Collection， 这 些 类 型 都 实现 了 用 来 添加 新 元 素 的 
add( 方 法 。 

ArrayList 和 LinkedList 都 是 List 类 型 ， 从 输出 可 以 看 出 ， 它们 都 按照 被 插入 的 顺序 保存 元 素 。 
两 者 的 不 同 之 处 不 仅 在 于 执行 某 些 类 型 的 操作 时 的 性 能 ， 而 且 LinkedList 包 含 的 操作 也 多 于 
ArrayList。 这 些 将 在 本 章 后 续 部 分 更 详细 地 讨论 。 

HashSet、TreeSet 和 LinkedHashSet 都 是 Set 类 型 ， 输 出 显示 在 Set 中 ， 每 个 相同 的 项 只 有 保 
存 一 次 ， 但 是 输出 也 显示 了 不 同 的 Set 实 现存 储 元 素 的 方式 也 不 同 。HashSet 使 用 的 是 相当 复杂 
的 方式 来 存储 元 素 的 ， 这 种 方式 将 在 第 17 章 中 介绍 ， 此 刻 你 只 需要 知道 这 种 技术 是 最 快 的 获取 
元 素 方式 ， 因 此 ， 存 储 的 顺序 看 起 来 并 无 实际 意义 (通常 你 只 会 关心 某 事物 是 否 是 某 个 Set 的 成 
员 ， 而 不 会 关心 它 在 Set 出 现 的 顺序 )。 如 果 存储 顺序 很 重要 ， 那 么 可 以 使 用 TreeSet， 它 按照 比 
较 结果 的 升序 保存 对 象 ， 或 者 使 用 LinkedHashSet， 它 按照 被 添加 的 顺序 保存 对 象 。 

Map (也 被 称 为 关联 数组 ) 使 得 你 可 以 用 键 来 查找 对 象 ， 就 像 一 个 简单 的 数据 库 。 键 所 关 
联 的 对 象 称 为 值 。 使 用 Map 可 以 将 美国 州 名 与 其 首府 联系 起 来 ， 如 果 想 知道 Ohio 的 首府 ， 可 以 
将 Ohio 作 为 键 进行 查找 ， 几 乎 就 像 使 用 数组 下 标 一 样 。 正 由 于 这 种 行为 ， 对 于 每 一 个 键 ，Map 
只 接受 存储 一 次 。 

Map.put(key,value) 方 法 将 增加 一 个 值 〈 你 想 要 增加 的 对 象 ) ， 并 将 它 与 某 个 键 (你 用 来 查 
找 这 个 值 的 对 象 ) 关联 起 来 。Map.get(key) 方 法 将 产生 与 这 个 键 相 关联 的 值 。 上 面 的 示例 只 添加 
了 键 - 值 对 ， 并 没有 执行 查找 ， 这 将 在 稍 后 展示 。 

注意 ， 你 不 必 指 定 (RAE) Map 的 尺寸 ， 因 为 它 自己 会 自动 地 调整 尺寸 。Map 还 知道 如 
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何 打印 自己 ， 它 会 显示 相关 联 的 键 和 值 。 键 和 值 在 Map 中 的 保存 顺序 并 不 是 它们 的 插入 顺序 ， 
因为 HashMap 实 现 使 用 的 是 一 种 非常 快 的 算法 来 控制 顺序 。 

本 例 使 用 了 三 种 基本 风格 的 Map: HashMap、TreeMap 和 LinkedHashMap。 与 HashSet 一 
样 ，HashMap 也 提供 了 最 快 的 查找 技术 ， 也 没有 按照 任何 明显 的 顺序 来 保存 其 元 素 。TreeMap 
按照 比较 结果 的 升序 保存 键 ， 而 LinkedHashMap 则 按照 插入 顺序 保存 键 ， 同 时 还 保留 了 
HashMap 的 查询 速度 。 

练习 4: (3) 创建 一 个 生成 器 类 ， 它 可 以 在 每 次 调用 其 next0 方 法 时 ， 产 生 你 由 你 最 喜欢 的 电 
影 (你 可 以 使 用 Snow White 或 Star Wars) 的 字符 构成 的 名 字 (作为 String 对 象 ) 。 在 字符 列表 中 
的 电影 名 用 完 之 后 ， 循 环 到 这 个 字符 列表 的 开始 处 。 使 用 这 个 生成 器 来 填充 数组 、ArrayList、 
LinkedList、HashSet、LinkedHashSet 和 TreeSet， 然 后 打印 每 一 个 容器 。 


11.5 List 


List 承 诺 可 以 将 元 素 维护 在 特定 的 序列 中 。List 接 口 在 Collection 的 基础 上 添加 了 大 量 的 方法 ， 
使 得 可 以 在 List 的 中 间 插 入 和 移 除 元 素 。 

有 两 种 类 型 的 List: 

"基本 的 ArrayList， 它 长 于 随机 访问 元 素 ， 但 是 在 List 的 中 间 插入 和 移 除 元 素 时 较 慢 。 

“LinkedList， 它 通过 代价 较 低 的 在 List 中 间 进行 的 插入 和 删除 操作 ， 提 供 了 优化 的 顺序 访 

问 。LinkedList 在 随机 访问 方面 相对 比较 慢 ， 但 是 它 的 特性 集 较 ArrayList 更 大 。 

下 面 的 示例 通过 导入 typeinfo.pets， 超 前 使 用 了 第 14 章 中 的 类 库 。 这 个 类 库 包含 了 Pet 类 的 继 
承 层次 结构 ， 以 及 用 于 随机 生成 Pet 对 象 的 一 些 工具 类 。 此 时 ， 你 还 不 需要 了 解 这 个 类 库 的 全 部 
内 容 ， 而 只 需要 知道 两 点 (D0) 有 一 个 Pet 类 ， 以 及 Pet 的 各 种 子 类 型 ，(2) 静 态 的 Pets.arrayList0 方 
法 将 返回 一 个 填充 了 随机 选取 的 Pet 对 象 的 ArrayList; 


//: hotding/ListFeatures.java 
import typeinfo.pets.*; 
import java.util.*; 
import static net.mindview.util.Print.*; 
public class ListFeatures { 
public static void main(String[] args) { 
Random rand = new Random(47) ; 
List<Pet> pets = Pets.arrayList(7); 
print("1: ”+ pets); 
Hamster h = new Hamster(); 
pets.add(h); // Automatically resizes 
print("2: " +-pets); 
print("3: ”+ pets.contains(h)); 
pets.remove(h); // Remove by object 
Pet p = pets.get(2); 
print("4: "+ p+" "+ pets. indexOf(p)): 
Pet cymric = new Cymric(); 
* + pets. indexOf(cymric)); 
”+ pets. remove(cymric)); 
the exact object: 
* + pets.remove(p)): 







List<Pet> sub = pets.subList(1, 4); 
print("subList: * + sub); 

print("19: ”+ pets.containsAll(sub)); 
Collections.sort(sub); // In-place sort 
print("sorted subList: " + sub); 

1/ Order is not important in containsAll(): 
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print("11: " + pets.containsAll(sub)); 
Collections. shuffle(sub, rand); // Mix it up 
print("shuffled subList: " + sub); 
print("12: ”+ pets.containsAll(sub)); 

List<Pet> copy = new ArrayList<Pet>(pets); 
sub = Arrays. asList(pets.get(1), pets.get(4)): 
print("sub: " + sub); 
copy.retainAll (sub): 
print("13: ”+ copy): 
copy = new ArrayList<Pet>(pets): // Get a fresh copy 
copy.remove(2); // Remove by index 
print("14: ”+ copy); 
copy.removeAll(sub): // Only removes exact objects 
print("15: " + copy); 
copy.set(1, new House()): // Replace an element 
print("16: " + copy): 
copy.addAll(2, sub); // Insert a list in the middle 
print("17: " + copy); 
print("18: ”+ pets.isEmpty()); 
pets.clear(); // Remove all elements 
print("19: ”+ pets); 
print("20: " + pets. isEmpty()); 
pets.addAll (Pets. arraylist(4)); 
print("21; ”+ pets); 
Object{] o = pets. toArray(); 
print("22: " + of3]); 
Pet(} pa = pets.toArray(new Pet[9]); 
print("23: " + paf3] .id()); 
} 
} /* Output: 
1: (Rat, Manx, Cymric, Mutt, Pug, Cymric, Pug) 
2: (Rat, Manx, Cymric, Mutt, Pug, Cymric, Pug, Hamster] 
3: true 
4: Cymric 2 
5: -1 r 
6: false 
7: true 
8: [Rat, Manx, Mutt, Pug, Cymric, Pug] 
9: [Rat, Manx, Mutt, Mouse, Pug, Cymric, Pug] 


subList: (Manx, Mutt, Mouse] 


10: 


true 


sorted subList: (Manx, Mouse, Mutt) 


il: 


true 


shuffled subList: [Mouse, Manx, Mutt] 


12: 


true 

[Mouse, Pug] 

[Mouse, Pug] 

{Rat, Mouse, Hutt, Pug, Cymric, Pug] 
{Rat, Mutt, Cymric, Pug] 

[Rat, Mouse, Cymric, Pug] 

[Rat, Mouse, Mouse, Pug. Cymric, Pug] 





(Manx, Cymric, Rat, EgyptianMau) 
EgyptianMau 
14 


打印 行 都 编 了 号 ， 因 此 输出 可 以 与 源码 相关 。 第 一 行 输出 展示 了 最 初 的 由 Pet 构成 的 List。 
与 数组 不 同 ，List 人 允许 在 它 被 创建 之 后 添加 元 素 、 移 除 元 素 ， 或 者 自我 调整 尺寸 。 这 正 是 它 的 重 


要 价值 所 在 





一 种 可 修改 的 序列 。 你 可 以 在 输出 行 2 中 看 到 添加 一 个 Hamster 的 结果 ， 即 对 象 被 


追加 到 了 表 尾 。 
你 可 以 用 contains0 方 法 来 确定 某 个 对 象 是 否 在 列表 中 。 如 果 你 想 移 除 一 个 对 象 ， 则 可 以 将 
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这 个 对 象 的 引用 传递 给 remove0 方 法 。 同 样 ， 如 果 你 有 一 个 对 象 的 引用 ， 则 可 以 使 用 indexOfO 
来 发 现 该 对 象 在 List 中 所 处 位 置 的 索引 编号 ， 就 像 你 在 输出 行 4 中 所 见 一 样 。 

当 确 定 一 个 元 素 是 否 属 于 某 个 List， 发 现 某 个 元 素 的 索引 ， 以 及 从 某 个 List 中 移 除 一 个 元 素 
时 ， 都 会 用 到 equals0 方 法 〈 它 是 根 类 Object 的 一 部 分 )。 每 个 Pet 都 被 定义 为 唯一 的 对 象 ， 因 此 
即使 在 列表 中 已 经 有 两 个 Cymric， 如 果 我 再 新 创建 一 个 Cymric， 并 把 它 传递 给 indexOf( 方 法 ， 
其 结果 仍 会 是 -1 (表示 未 找到 它 ) ， 而 且 尝 试 调用 remove0 方 法 来 删除 这 个 对 象 ， 也 会 返回 false。 
对 于 其 他 的 类 ，equals0 的 定义 可 能 有 所 不 同 。 例 如 ， 两 个 String 只 有 在 内 容 完全 一 样 的 情况 下 
才 会 是 等 价 的 。 因 此 为 了 防止 意外 ， 就 必须 意识 到 List 的 行为 根据 equals0 的 行为 而 有 所 变化 。 

在 输出 行 7 和 8 中 ， 展 示 了 对 精确 匹配 List 中 某 个 对 象 的 对 象 进行 移 除 是 成 功 的 。 

在 List 中 间 插 入 元 素 是 可 行 的 ， 就 像 你 在 输出 行 9 和 它 前 面 的 代码 中 所 看 到 的 一 样 。 但 是 这 
带 来 了 一 个 问题 : 对 于 LinkedList， 在 列表 中 间 插 入 和 删除 都 是 廉价 操作 (在 本 例 中 ， 除 了 对 列 
表 中 间 进 行 的 真正 的 随机 访问 ) ， 但 是 对 于 ArrayList， 这 可 是 代价 高 昂 的 操作 。 这 是 否 意味 着 
你 应 该 永远 都 不 要 在 ArrayList 的 中 间 插 入 元 素 ， 并 最 好 是 切换 到 LinkedList? 不 ， 这 仅仅 意味 
着 ， 你 应 该 意识 到 这 个 问题 ， 如 果 你 开始 在 某 个 ArrayList 的 中 间 执行 很 多 插入 操作 ， 并 且 你 的 
程序 开始 变 慢 ， 那 么 你 应 该 看 看 你 的 List 实 现 有 可 能 就 是 罪魁 祸首 (发 现 此 类 瓶颈 的 最 佳 方式 是 
使 用 仿真 器 ， 就 像 你 在 http://MindView.netWBooks/BetterJava 上 的 补充 材料 中 所 看 到 的 一 样 )。 优 
化 是 一 个 很 塘 手 的 问题 ， 最 好 的 策略 就 是 置 之 不 顾 ， 直 到 你 发 现 需 要 担心 它 了 (尽管 理解 这 些 
问题 总 是 一 种 好 的 思路 ) 。 

subList0 方 法 允许 你 很 容易 地 从 较 大 的 列表 中 创建 出 一 个 片断 ， 而 将 其 结果 传递 给 这 个 较 
大 的 列表 的 containsAlI0 方 法 时 ， 很 自然 地 会 得 到 true。 还 有 一 点 也 很 有 趣 ， 那 就 是 我 们 注意 到 
顺序 并 不 重要 ， 你 可 以 在 输出 行 11 和 12 中 看 到 ， 在 sub 上 调用 名 字 很 直观 的 Colleetions.sort0 和 
Collection.shuffle0 方 法 ， 不 会 影响 containsAll0 的 结果 。subList0 所 产生 的 列表 的 幕后 就 是 初始 
列表 ， 因 此 ， 对 所 返回 的 列表 的 修改 都 会 反映 到 初始 列表 中 ， 反 之 亦 然 。 

retainAlI0 方 法 是 一 种 有 效 的 “交集 ”操作 ， 在 本 例 中 ， 它 保留 了 所 有 同时 在 copy 与 sub 中 
的 元 素 。 请 再 次 注意 ， 所 产生 的 行为 依赖 于 equals0 方 法 。 

输出 行 14 展 示 了 用 元 素 的 索引 值 来 移 除 元 素 的 结果 。 与 通过 对 象 引用 来 移 除 相 比 ， 它 显得 
更 加 直观 ， 因 为 在 使 用 索引 值 时 ， 不 必 担 心 equais0 的 行为 。 

removeAli0 方 法 的 行为 也 是 基于 equals0 方 法 的 。 就 像 其 名 称 所 表示 的 ， 它 将 从 List 中 移 除 
在 参数 List 中 的 所 有 元 素 。 

set0 方 法 的 命名 显得 很 不 合 时 宜 ， 因 为 它 与 Set 类 存在 潜在 的 冲突 。 在 此 处 ，replace 可 能 会 显 
得 更 适合 ， 因 为 它 的 功能 是 在 指定 的 索引 处 (第 一 个 参数 ) ， 用 第 二 个 参数 替换 整个 位 置 的 元 素 。 

输出 行 17 表 明 ， 对 于 List， 有 一 个 重 载 的 addAll0 方 法 使 得 我 们 可 以 在 初始 List 的 中 间 插入 
新 的 列表 ， 而 不 仅仅 只 能 用 Collection 中 的 addAli0 方 法 将 其 追加 到 表 尾 。 

输出 行 18-20 展 示 了 isEmpty0 和 elear0 方 法 的 效果 。 

输出 行 22-23 展 示 了 你 可 以 如 何 通 过 使 用 toArray0 方 法 ， 将 任意 的 Collection 转 换 为 一 个 数 
组 。 这 是 一 个 重 载 方 法 ， 其 无 参数 版 本 返回 的 是 Objeet 数 组 ， 但 是 如 果 你 向 这 个 重 载 版 本 传递 
目标 类 型 的 数据 ， 那 么 它 将 产生 指定 类 型 的 数据 (假设 它 能 通过 类 型 检查 ) 。 如 果 参 数 数组 太 小 ， 
存放 不 下 List 中 的 所 有 元 素 (就 像 本 例 一 样 )，toArray0 方 法 将 创建 一 个 具有 合适 尺寸 的 数组 。 
Pet 对 象 具有 一 个 ia0 方 法 ， 如 你 所 见 ， 可 以 在 所 产生 的 数组 中 的 对 象 上 调用 这 个 方法 。 

练习 5，(3) 修改 ListFeatures.java， 让 它 使 用 Integer ( 记 住 自动 包装 机 制 ! ) 而 不 是 Pet， 
并 解释 在 结果 上 有 何不 同 。 
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练习 6: (2) 修改 ListFeaturesjava， 让 它 使 用 String 而 不 是 Pet， 并 解释 在 结果 上 有 何不 同 。 
练习 7: (3) 创建 一 个 类 ， 然 后 创建 一 个 用 你 的 类 的 对 象 进行 过 初始 化 的 数组 。 通 过 使 用 
subList0 方 法 ， 创 建 你 的 List 的 子 集 ， 然 后 在 你 的 List 中 移 除 这 个 子 集 。 


11.6 迭代 器 


任何 容器 类 ， 都 必须 有 某 种 方式 可 以 插入 元 素 并 将 它们 再 次 取 回 。 毕 竟 ， 持 有 事物 是 容器 
最 基本 的 工作 。 对 于 List，add0 是 插入 元 素 的 方法 之 一 ， 而 get0 是 取出 元 素 的 方法 之 一 。 

如 果 从 更 高 层 的 角度 思考 ， 会 发 现 这 里 有 个 缺点 ， 要 使 用 容器 ， 必 须 对 容器 的 确切 类 型 编 
程 。 初 看 起 来 这 没什么 不 好 ， 但 是 考虑 下 面 的 情况 : 如 果 原 本 是 对 着 List 编 码 的 ， 但 是 后 来 发 现 
如 果 能 够 把 相同 的 代码 应 用 于 Set， 将 会 显得 非常 方便 ， 此 时 应 该 怎么 做 ? 或 者 打算 从 头 开始 编 
写 通 用 的 代码 ， 它 们 只 是 使 用 容器 ， 不 知道 或 者 说 不 关心 容器 的 类 型 ， 那 么 如 何 才能 不 重 写 代 
码 就 可 以 应 用 于 不 同类 型 的 容器 ? 

BRE (也 是 一 种 设计 模式 ) 的 概念 可 以 用 于 达成 此 目的 。 和 迭代 器 是 一 个 对 象 ， 它 的 工作 
是 遍历 并 选择 序列 中 的 对 象 ， 而 客户 端 程序 员 不 必 知 道 或 关心 该 序列 底层 的 结构 。 此 外 ， 和 友 代 
器 通常 被 称 为 轻 量 级 对 象 : 创建 它 的 代价 小 。 因 此 ， 经 常 可 以 见 到 对 迭代 器 有 些 奇怪 的 限制 ， 
例如 ，Java 的 Iterator 只 能 单 向 移动 ， 这 个 Iterator 只 能 用 来 : 

1) 使 用 方法 iterator0 要 求 容器 返回 一 个 Iterator。Iterator 将 准备 好 返回 序列 的 第 一 个 元 素 。 

2) 使 用 next0 获 得 序列 中 的 下 一 个 元 素 。 

3) 使 用 hasNext0 检 查 序列 中 是 否 还 有 元 素 。 

4) 使 用 remove0 将 迭代 器 新 近 返 回 的 元 素 删 除 。 

为 了 观察 它 的 工作 方式 ， 让 我 们 再 次 使 用 在 第 14 章 中 的 Pet 工具 : 


//: hotdtng/SimpteIteration .java 
import typeinfo.pets.*; 
import java.util.*; 


public class Simplelteration { 
public static void main(String{] args) { 
List<Pet> pets = Pets.arrayList (12); 
Iterator<Pet> it = pets.iterator(); 
while(it.hasNext()) { 
Pet p = it.next(); 
System.out.print(p.id() + ":" +p +" "); 


System.out.printin(); 
// A simpler approach, when possible: 
for(Pet p : pets) 

System.out.print(p.id() + ":" +p +" "); 
System. out.printin(): 
// An Iterator can also remove elements: 
it = pets. iterator (); 
for(int i = @; 1 <6; i++) { 

it.next(); 

it.remove(); 


} 
System.out .printin(pets); 


} 
} /* Output: 
@:Rat 1:Manx 2:Cymric 3:Mutt 4:Pug 5:Cymric 6:Pug 7:Manx 
8:Cymric 9:Rat 10:EgyptianMau 11:Hamster 
@:Rat 1:Manx 2:Cymric 3:Mutt 4:Pug 5:Cymric 6:Pug 7:Manx 
8:Cymric 9:Rat 10:EgyptianMau 11:Hamster 
[Pug, Manx, Cymric, Rat, EgyptianMau, Hamster} 
Wh 
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有 了 Iterator 就 不 必 为 容器 中 元 素 的 数量 操心 了 ， 那 是 由 hasNext0 和 next0 关 心 的 事情 。 

如 果 你 只 是 向 前 遍历 List， 并 不 打算 修改 List 对 象 本 身 ， 那 么 你 可 以 看 到 foreaeh 语 法 会 显得 
ee ees 这 意味 着 在 调用 remove0 之 前 必须 先 调用 
oe Luu ANNA: Indes LCA RANE ES 并 且 侦 穿 于 
ia E “EAL ames BIOTIN. 


//: holding/CrossContainerIteration. java 
import typeinfo.pets.*; 
import java.util.*; 


public class CrossContainerlIteration { 
public static void display(Iterator<Pet> it) { 
while(it.hasNext()) { 
Pet p = it.next(); 
System.out.print(p.id() + ":"+#p+" "); 


System. out.printin(); 


} 

public static void main(String[] args) { 
ArrayList<Pet> pets = Pets.arrayList(8); 
LinkedList<Pet> petsLL = new LinkedList<Pet>(pets); 
HashSet<Pet> petsHS = new HashSet<Pet>(pets) ; 
TreeSet<Pet> petsTS = new TreeSet<Pet>(pets) ; 
display (pets. iterator()); 
display (petsLL.iterator()); 
display (petsHS.iterator()): 
display (petsTS. iterator()); 


} 
} /* Output: 
6:Rat 1:Manx 2:Cymric 3:Mutt 4:Pug 5:Cymric 6:Pug 7:Manx 
t 1:Manx 2:Cymric 3:Mutt 4:Pug 5:Cymric 6:Pug 7:Manx 
4:Pug 6:Pug 3:Mutt 1:Manx 5:Cymric 7:Manx 2:Cymric @:Rat 
S:Cymric 2:Cymric 7:Manx 1:Manx 3:Mutt 6:Pug 4:Pug @:Rat 
Wm 


请 注意 ，display0 方 法 不 包含 任何 有 关 它 所 遍历 的 序列 的 类 型 信息 ， 而 这 也 展示 了 Iterator 
的 真正 威力 : 能 够 将 遍 爵 序列 的 操作 与 序列 底层 的 结构 分 离 。 正 由 于 此 ， 我 们 有 时 会 说 : 迭代 
器 统一 了 对 容器 的 访问 方式 。 

练习 8: (1) 修改 练习 1， 以 便 调 用 hop0 时 使 用 Iterator 人 遍历 List。 

练习 9，(4) 修改 innerclasses/Sequencejava， 使 得 在 Sequence 中 ， 用 Iterator 取 代 Selector。 

练习 10: (2) 修改 第 8 章 中 的 练习 9， 使 其 使 用 一 个 ArrayList 来 存放 Rodents， 并 使 用 一 个 
Iterator 来 访问 Rodent 序 列 。 

练习 11: (2) 写 一 个 方法 ， 使 用 Iterator 人 遍历 Collection， 并 打印 容器 中 每 个 对 象 的 toString0 。 
填充 各 种 类 型 的 Collection， 然 后 对 其 使 用 此 方法 。 
11.6.1 Listiterator 

ListIterator 是 一 个 更 加 强大 的 Iterator 的 子 类 型 ， 它 只 能 用 于 各 种 List 类 的 访问 。 尽 管 
Iterator 只 能 向 前 移动 ， 但 是 ListIterator 可 以 双向 移动 。 它 还 可 以 产生 相对 于 和 迭代 器 在 列表 中 指 





日 remove0 是 所 谓 的 “可 选 ” 方 法 (还 有 一 些 其 他 的 这 种 方法 )， 即 不 是 所 有 的 Tterator 实 现 都 必须 实现 该 方法 。 
这 个 问题 将 在 第 17 章 中 介绍 。 但 是 ， 标 准 Java 类 库 实现 了 remove0， 因 此 直至 第 17 章 之 前 ， 你 都 不 用 担心 这 个 
问题 。 
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向 的 当前 位 置 的 前 一 个 和 后 一 个 元 素 的 索引 ， 并 且 可 以 使 用 set0 方 法 替换 它 访问 过 的 最 后 一 个 
元 素 。 你 可 以 通过 调用 listIterator0 方 法 产生 一 个 指向 List 开 始 处 的 ListIterator， 并 且 还 可 以 通 
过 调用 listIterator(n) 方 法 创建 一 个 一 开始 就 指向 列表 索引 为 a 的 元 素 处 的 ListIterator。 下 面 的 示 
例 演示 了 所 有 这 些 能 力 : 


//: hotding/ListIteration.java 
import typeinfo.pets.*; 
import java.util.*; 


public class ListIteration { 

Public static void main(String() args) { 
ListsPet> pets = Pets. arrayList (8): 
ListIterator<Pet> it = pets. listIterator(); 
while(it.hasNext()) 

System. out.print(it.next() +", “+ it.nextIndex() + 

", "+ it.previousIndex() + "; "): 

System.out.printin(): 
// Backwards: 
while(it.hasPrevious()) 

System. out. print (it.previous().id() +" "); 
System.out.printin(); 
System. out.printin(pets); 
it = pets. listIterator (3); 
while(it.hasNext()) { 

it.next(); 

it.set(Pets.randomPet()): 


System. out.printin(pets); 


} 
} /* Output: 
Rat, 1, 8; Manx, 2, 1; Cymric, 3, 2; Mutt, 4, 3; Pug, 5, 4: 
Cymric, 6, 5; Pug, 7, 6; Manx, 8, 7; 
76543216 
IRat, Manx, Cymric, Mutt. Pug, Cymric, Pug, Manx] 
(Rat, Manx, Cymric, Cymric, Rat, EgyptianMau, Hamster, 
EgyptianMau] 
“ 


li~ 
PetrandomPet( 方 法 用 来 替换 在 列表 中 从 位 置 3 开始 向 前 的 所 有 Pet 对 象 。 
练习 12: (3) 创建 并 组 装 一 个 List<Integer>， 然 后 创建 第 二 个 具有 相同 尺寸 的 List<Integer> ， 
并 使 用 ListIterator 读 取 第 一 个 List 中 的 元 素 ， 然 后 再 将 它们 以 反 序 插入 到 第 二 个 列表 中 (你 可 能 
想 探 索 该 问题 的 各 种 不 同 的 解决 之 道 )。 


11.7 LinkedList 


LinkedList 也 像 ArrayList 一 样 实现 了 基本 的 List 接 口 ， 但 是 它 执行 某 些 操作 (在 List 的 中 间 
插入 和 移 除 ) 时 比 ArrayList 更 高 效 ， 但 在 随机 访问 操作 方面 却 要 逊色 一 些 。 

LinkedList 还 添加 了 可 以 使 其 用 作 栈 、 队 列 或 双 端 队列 的 方法 。 

这 些 方法 中 有 些 彼此 之 间 只 是 名 称 有 些 差异 ， 或 者 只 存在 些许 差异 ， 以 使 得 这 些 名 字 在 特 
定 用 法 的 上 下 文 环境 中 更 加 适用 《特别 是 在 Queue 中 )。 例 如 ，getFirst0 和 element0 完 全 一 样 ， 
它们 都 返回 列表 的 头 《 第 一 个 元 素 )， 而 并 不 移 除 它 ， 如 果 List 为 空 ， 则 抛 出 NoSuchElement- 
Exeeption。peek0 方 法 与 这 两 个 方式 只 是 稍 有 差异 ， 它 在 列表 为 空 时 返回 null。 

removeFirst0 与 remove0 也 是 完全 一 样 的 ， 它 们 移 除 并 返回 列表 的 头 ， 而 在 列表 为 空 时 抛 出 
NoSuchElementException。polli0 稍 有 差异 ， 它 在 列表 为 空 时 返回 null。 

addFirst0 与 add0O 和 addLast0 相 同 ， 它 们 都 将 某 个 元 素 插 入 到 列表 的 尾 ( 端 ) 部 。 

removeLast0 移 除 并 返回 列表 的 最 后 一 个 元 素 。 
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下 面 的 示例 展示 了 这 些 特性 之 间 基本 的 相同 性 和 差异 性 ， 它 重复 地 执行 ListFeatures.java 中 
所 示 的 行为 : 

//: holding/LinkedListFeatures. java 

import typeinfo.pets.* 


import java.util.* 
import static net.mindview.util.Print.*; 





public class LinkedListFeatures { 
public static void main(String[] args) { 
LinkedList<Pet> pets = 
new LinkedList<Pet>(Pets.arrayList(5)); 

print (pets) ; 
// Identical: 
print("pets.getFirst(): ”+ pets.getFirst()); 
Print(*pets.element(): ”+ pets.element()); 
// Only differs in empty-list behavior: 
print("pets.peek(): ”+ pets.peek()); 
// Identical; remove and return the first element: 
print("pets.remove(): ”+ pets.remove()); 
print("pets.removeFirst(): ”+ pets.removeFirst()); 
// Only differs in empty-list behavior: 
print("pets.poll(): " + pets.poll()); 
print (pets); 
pets.addFirst (new Rat()); 
print("After addFirst(): " + pets); 
pets.offer(Pets.randomPet()); 
print("After offer(): ”+ pets); 
pets.add(Pets.randomPet ()) ; 
print("After add(): ”+ pets); . 
pets.addLast (new Hamster ()) ; 
print(*After addLast(): ”+ pets): 
print("pets.removelast(): ”+ pets.removeLast()); 


} 
} /* Output: 
(Rat, Manx, Cymric, Mutt, Pug] 
pets.getFirst(): Rat 
pets.element(): Rat 
pets.peek(): Rat 
pets.remove(): Rat 
pets.removeFirst(): Manx 
pets.poll(): Cymric 
(Mutt, Pug) 
After addFirst(): (Rat, Mutt, Pug] 
After offer(): (Rat, Mutt, Pug. Cymric) 
After add(): (Rat, Mutt, Pug. Cymric, Pug} 
After addLast(): [Rat, Mutt, Pug. Cymric, Pug, Hamster] 
pets.removeLast(): Hamster 
Whim 


Pets.arrayListO 的 结果 交 给 了 LinkedList 的 构造 器 ， 以 便 使 用 它 来 组 装 LinkedList。 如 果 
你 浏览 一 下 Queue 接 口 就 会 发 现 ， 它 在 LinkedList 的 基础 上 添加 了 elementO、offer0、peek0O、 
pollO 和 remove( 方 法 ， 以 使 其 可 以 成 为 一 个 Queue 的 实现 。Queue 的 完整 示例 将 在 本 章 稍 后 


给 出 。 

练习 13: (3) 在 innerclasses/GreenhouseController.java 示 例 中 ，Controller 类 使 用 的 是 
ArrayList， 修 改 代码 ， 用 LinkedList 替 换 之 ， 并 使 用 Iterator 来 循环 遍历 事件 集 。 

练习 14: (3) 创建 一 个 空 的 LinkedList<Integer>， 通 过 使 用 ListIterator， 将 若干 个 Integer 插 
入 这 个 List 中 ， 插 入 时 ， 总 是 将 它们 插入 到 List 的 中 间 。 


11.8 Stack 
“ 栈 ” 通 常 是 指 “ 后 进 先 出 ”(LIFO) WAZ. AMR AeA, HAR “EA” 
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栈 的 元 素 ， 第 一 个 “弹出 ” 栈 。 经 常用 来 类 比 栈 的 事物 是 装 有 弹簧 的 储 放 器 中 的 自助 餐 托盘 ， 
最 后 装 和 的 托盘 总 是 最 先 拿 出 使 用 的 。 

LinkedList 具 有 能 够 直接 实现 栈 的 所 有 功能 的 方法 ,因此 可 以 直接 将 LinkedList 作 为 栈 使 用 。 
不 过 ， 有 了 时 一 个 真正 的 “ 栈 ”更 能 把 事情 讲 清楚 : 


/1/1: net/mindview/util/Stack. java 

// Making a stack from a LinkedList. 
package net.mindview.util; 

import java.util.LinkedList; 


public class Stack<T> { 
private LinkedList<T> storage = new LinkedList<T>(); 
public void push(T v) { storage.addFirst(v); } 
public T peek() { return storage.getFirst(); } 
public T pop() { return storage.removeFirst(); } 
public boolean empty() { return storage. isEmpty(): } 
public String toString() { return storage.toString(); } 
} Mi~ 


这 里 通过 使 用 范 型 ， 引 入 了 在 栈 的 类 定义 中 最 简单 的 可 行 示例 。 类 名 之 后 的 <T> 告 诉 编译 
器 这 将 是 一 个 参数 化 类 型 ， 而 其 中 的 类 型 参数 ， 即 在 类 被 使 用 时 将 会 被 实际 类 型 替换 的 参数 ， 
就 是 T。 大 体 上 ， 这 个 类 是 在 声明 “我 们 在 定义 一 个 可 以 持 有 T 类 型 对 象 的 Stack。”Stack 是 用 
LinkedList 实 现 的， 而 LinkedList 也 被 告知 它 将 持 有 T 类 型 对 象 。 注 意 ，push0 接 受 的 是 T 类 型 的 
对 象 ， 而 peek0 和 pop0 将 返回 T 类 型 的 对 象 。peek() 方 法 将 提供 栈 顶 元 素 ， 但 是 并 不 将 其 从 栈 顶 
移 除 ， 而 pop0 将 移 除 并 返回 栈 顶 元 素 。 

如 果 你 只 需要 栈 的 行为 ， 这 里 使 用 继承 就 不 合适 了 ， 因 为 这 样 会 产生 具有 LinkedList 的 其 他 
所 有 方法 的 类 〈 就 象 你 将 在 第 17 章 中 所 看 到 的 ，Javal.0 的 设计 者 在 创建 java.util.Stack 时 ， 就 犯 
了 这 个 错误 ) 。 

下 面 演示 了 这 个 新 的 Stack 类 : 


/1/: holding/StackTest.java 
‘import net.mindview.util.*; 


public class StackTest { 
public static void main(String{] args) { 
Stack<String> stack = new Stack<String>(); 
for(String s : "My dog has fleas".split(" ")) 
stack. push(s); 
while(!stack.empty()) 
System.out.print(stack.pop() + " "); 


} 
} /* Output: 
fleas has dog My 
Whim 


如 果 你 想 在 自己 的 代码 中 使 用 这 个 Stack 类 ， 当 你 在 创建 其 实例 时 ， 就 需要 完整 指定 包 名 ， 
或 者 更 改 这 个 类 的 名 称 ， 否 则 ， 就 有 可 能 与 java.util 包 中 的 Stack 发 生 冲 突 。 例 如 ， 如 果 我 们 在 
上 面 的 例子 中 导入 java.util.*， 那 么 就 必须 使 用 包 名 以 防止 冲突 : 


//: holding/StackCollision. java 
import net.mindview.util.*: 


public class StackCollision { 
public static void main(String{] args) { 
net .mindview.util.Stack<String> stack = 
new net .mindview.util.Stack<String>() ; 
for(String s : “My dog has fleas".split(" *)) 
stack. push(s) ; 
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while(!stack.empty()) 
System.out.print(stack.pop() + " "); 
System.out.printin(): 
java.util.Stack<String> stack2 = 
new java.util.Stack<String>(); 
for (String s : "My dog has fleas*.split(” ")) 
stack2.push(s) ; 
while(!stack2.empty()) 
System.out.print (stack2.pop() + " *); 


} 
} /* Output: 
fleas has dog My 
fleas has dog My 
Wh 


这 两 个 Stack 具 有 相同 的 接口 ， 但 是 在 java.atil 中 没有 任何 公共 的 Stack 接 口 ， 这 可 能 是 因为 
在 Javal.0 中 的 设计 欠 佳 的 最 初 的 java.atil.Stack 类 占用 了 这 个 名 字 。 尽管 已 经 有 了 javautil.Stack， 
但 是 LinkedList 可 以 产生 更 好 的 Stack， 因 此 net.mindview.util.Stack 所 采用 的 方式 更 是 可 取 的 。 

你 还 可 以 通过 显 式 的 导入 来 控制 对 “首选 ”Stack 实 现 的 选择 ， 

import net.mindview.util. Stack: 

现在 ， 任 何 对 Stack 的 引用 都 将 选择 netmindview.ntil 版 本 ， 而 在 选择 java.util.Stack 时 ， 必 
须 使 用 全 限定 名 称 。 

练习 15，(4) 栈 在 编程 语言 中 经 常用 来 对 表达 式 求 值 。 请 使 用 net.mndview.util.Stack 对 下 面 的 
表达 式 求 值 ， 其 中 “+” 表 示 “ 将 后 面 的 字母 压 进 栈 "， 而 “-- ”表示 “弹出 栈 顶 字母 并 打印 它 "; 


“++U+n+C---+e+I+t--- 十 3- 十 -十 D+t+y--- 十 -+I+U-- 十 ]+e+S- 一 ” 
11.9 Set 


Set 不 保存 重复 的 元 素 (至 于 如 何 判断 元 素 相同 则 较为 复杂 ， 稍 后 便 会 看 到 ) 。 如 果 你 试图 
将 相同 对 象 的 多 个 实例 添加 到 Set 中 ， 那 么 它 就 会 阻止 这 种 重复 现象 。Set 中 最 常 被 使 用 的 是 测试 
归属 性 ， 你 可 以 很 容易 地 询问 某 个 对 象 是 否 在 某 个 Set 中 。 正 因 如 此 ， 查 找 就 成 为 了 Set 中 最 重要 
的 操作 ， 因 此 你 通常 都 会 选择 一 个 HashSet 的 实现 ， 它 专门 对 快速 查找 进行 了 优化 。 

Set 具 有 与 Collection 完 全 一 样 的 接口 ， 因 此 没有 任何 额外 的 功能 ， 不 像 前 面 有 两 个 不 同 的 
List。 实 际 上 Set 就 是 Collection， 只 是 行为 不 同 。( 这 是 继承 与 多 态 思想 的 典型 应 用 ， 表 现 不 同 
的 行为 。) Set 是 基于 对 象 的 值 来 确定 归属 性 的 ， 而 更 加 复杂 的 问题 我 们 将 在 第 17 章 中 介绍 。 

下 面 是 使 用 存放 Integer 对 象 的 HashSet 的 示例 : 

/1/: holding/SetOfInteger .java 

import java.util.*; 


public class SetOfInteger { 
public static void main(String{] args) { 
Random rand = new Random(47) ; 
Set<Integer> intset = new HashSet<Integer>(): 
for(int i = 0; 1 < 10000; i++) 
intset.add(rand.nextInt(36)); 
System. out.printin(intset) ; 


} 
}/* Output: 
[15, 8, 23, 16, 7, 22, 9, 21, 6, 1,.29, 14, 24, 4, 19, 26, 
11, 18, 3, 12, 27, 17, 2, 13, 28, 26, 25, 10, 5, 6] 
Mim 


在 0 到 29 之 间 的 10000 个 随机 数 被 添加 到 了 Set 中 ， 因 此 你 可 以 想象 ， 每 一 个 数 都 重复 了 许多 
次 。 但 是 你 可 以 看 到 ， 每 一 个 数 只 有 一 个 实例 出 现在 结果 中 。 
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你 还 可 以 注意 到 ， 输 出 的 顺序 没有 任何 规律 可 循 ， 这 是 因为 出 于 速度 原因 的 考虑 ，HashSet 
使 用 了 散 列 一 一 散 列 将 在 第 17 章 中 介绍 。HashSet 所 维护 的 顺序 与 TreeSet 或 LinkedHashSet 都 不 
同 ， 因 为 它们 的 实现 具有 不 同 的 元 素 存储 方式 。TreeSet 将 元 素 存储 在 红 一 黑 树 数据 结构 中 ， 而 
HashSet 使 用 的 是 散 列 函数 。LinkedHashList 因 为 查询 速度 的 原因 也 使 用 了 散 列 ， 但 是 看 起 来 它 
使 用 了 链表 来 维护 元 素 的 插入 顺序 。 

如 果 你 想 对 结果 排序 ， 一 种 方式 是 使 用 TreeSet 来 代替 HashSet: 


//: holding/SortedSetOfInteger .java 
‘import java.util.*; 


public class SortedSetOfInteger ( 
Public static void main(String[] args) { 
Random rand = new Random(47); 
SortedSet<Integer> intset = new TreeSet<Integer>(); 
for(int i = @; 1 < 10008; i++) 
intset.add(rand.nextInt(30)): 
System.out.printin(intset); 


} 
} /* Output: 
(0, 1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 11, 12, 13, 14, 15, 16, 
17, 18, 19, 20, 21, 22, 23, 24, 25, 26, 27, 28, 29] 
Whim 


你 将 会 执行 的 最 常见 的 操作 之 一 ， 就 是 使 用 contains0 测 试 Set 的 归属 性 ， 但 是 还 有 很 多 操作 
会 让 你 想起 在 上 小 学 时 所 教授 的 文 氏 图 ( 译 者 注 : 用 圆 表示 集 与 集 之 间 关 系 的 图 ) : 


//: holding/SetOperations. java 
import java.util.*; 

import static net.mindview.util.Print.*; 
public class SetOperations { 

Public static void main(String{] args) { 
Set<String> setl = new HashSet<String>(); 
Collections. addAll(set1, 

"“ABCDEFGHIJKL".split(™ ")); 
setl.add("M") ; 
print("H: " + setl.contains("H")); 
print("N: ”+ setl.contains("N")); 
Set<String> set2 = new HashSet<String>(); 
Collections.addAll(set2, "H I J K L".split(’ ")); 
print("set2 in setl: " + set1.containsAll(set2)); 
seti.remove("H"); 
print("setl: " + set1); 
print("set2 in setl: ”+ set1.containsAll(set2)): 
set1.removeAll (set2) ; 
print("set2 removed from setl: " + setl); 
Collections. addAll(set1, "X Y Z".split(" ")); 
print("'X Y Z' added to setl: " + set1); 


} 
} /* Output: 
H: true 
N: false 
set2 in seti: true 
setl: [D, K, C, B, L, G, I, M, A, F, J, E] 
set2 in setl: false 
set2 removed from setl: [D. C, B, G, M. A, F, EJ 
'X Y Z' added to setl: [Z, D, C, B, G, M, A, F, Y, X, E] 
S i~ 


这 些 方法 名 都 是 自 解释 型 的 ， 而 有 几 个 方法 可 以 在 JDK 文 档 中 找到 。 

能 够 产生 每 个 元 素 都 唯一 的 列表 是 相当 有 用 的 功能 。 例 如 ， 在 你 想 要 列 出 在 上 面 的 
SetOperations.java 文 件 中 所 有 的 单词 的 时 候 。 通 过 使 用 本 书 稍 后 将 要 介绍 的 net.mindview. 
TextFile 工 具 ， 可 以 打开 一 个 文件 ， 并 将 其 读 入 一 个 Set 中 : 





[> 
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/1/: holding/UniqueWords. java 
import java.util. *; 4 
inport net.mindview.utit.*; 


public class UniqueWords { 
public static void main(String{] args) { 
Set<String> words = new TreeSet<String>( 
new TextFile("SetOperations. java", “\\W+")); 
System. out.printin(words) : 


} 
} /* Output: 
[A, B, C, Collections, D, E, F, G, H, HashSet, I, J, K, L, 
M, N, Output, Print, Set, SetOperations, String, X, Y, Z, 
add, addAll, added, args, class, contains, containsAll, 
false, from, holding, import, in, java, main, mindview, 
net, new, print, public, remove, removeAll, removed, setl, 
set2, split, static, to, true, util, void] 
Whim 


TextFile 继 承 自 List<String>， 其 构造 器 将 打开 文件 ， 并 根据 正则 表达 式 “WW+” 将 其 断 开 
为 单词 ， 这 个 正则 表达 式 表示 “一 个 或 多 个 字母 ”( 正 则 表达 式 将 在 第 13 章 中 介绍 )。 所 产生 的 
结果 传递 给 了 TreeSet 的 构造 器 ， 它 将 把 List 中 的 内 容 添加 到 自身 中 。 由 于 它 是 TreeSet， 因 此 其 
结果 是 排序 的 。 在 本 例 中 ， 排 序 是 按 字典 序 进行 的 ， 因 此 大 写 和 小 写字 母 被 划分 到 了 不 同 的 组 
中 。 如 果 你 想 要 按照 字母 序 排序 ， 那 么 可 以 向 TreeSet 的 构造 器 传 入 String.CASE_INSENTIVE_ 
ORDER 比较 器 (比较 器 就 是 建立 排序 顺序 的 对 象 ) : 

/7: holding/UniqueWordsAlphabetic. java 

// Producing an alphabetic Listing. 


import java.util.*; 
import net.mindview.util.*; 


public class UniqueWordsAlphabetic { 
public static void main(String[] args) { 
Set<String> words = 
new TreeSet<String>(String.CASE_INSENSITIVE_ORDER) ; 
words. addA11( 
new TextFile("SetOperations. java", "\\W+")); 
System. out.printtn(words) ; 


} 
} /* Output: 
[A, add, addAll, added, args, B, C, class, Collections, 
contains, containsAll, D, E, F, false, from, G, H, HashSet, 
holding, I, import, in, J, java, K, L, M, main, mindview, 
N, net, new, Output, Print, public, remove, removeAll, 
removed, Set, setl, set2, SetOperations, split, static, 
String, to, true, util, void, X, Y, Z) 
Whim 


Comparator 比 较 器 将 在 第 16 章 详细 介绍 。 
练习 16: (5) 创建 一 个 元 音字 母 Set。 对 UniqueWordsjava 操 作 ， 计 数 并 显示 在 每 一 个 输入 
单词 中 的 元 音字 母 数量 ， 并 显示 输入 文件 中 的 所 有 元 音字 母 的 数量 总 和 。 


11.10 Map 


将 对 象 映射 到 其 他 对 象 的 能 力 是 一 种 解决 编程 问题 的 杀手 铜 。 例 如 ， 考 虑 一 个 程序 ， 它 将 
用 来 检查 Java 的 Random 类 的 随机 性 。 理 想 状态 下 ，Random 可 以 将 产生 理想 的 数字 分 布 ， 但 要 
想 测试 它 ， 则 需要 生成 大 量 的 随机 数 ， 并 对 落 入 各 种 不 同 范围 的 数字 进行 计数 。Map 可 以 很 容 
易 地 解决 该 问题 。 在 本 例 中 ， 键 是 由 Random 产 生 的 数字 ， 而 值 是 该 数字 出 现 的 次 数 : 
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/1: holding/Statistics.java 
// Simple demonstration of HashMap. 
‘import java.util.*; 


public class Statistics { 
public static void main(String[] args) { 

Random rand = new Random(47); 
Map<Integer,Integer> m = 

new HashMap<Integer , Integer>(); 
for(int 1 = 9; i < 10008; i++) { 

// Produce a number between 9 and 20: 

int r = rand.nextInt(26) ; 

Integer freq = m.get(r); 

m.put(r, freq == null ? 1 : freq + 1); 





} 
System.out.printin(m); 
} 

} /* Output: 
{15=497, 4=481, 19=464, 8=468, 11=531, 16=533, 18=478, 
3=508, 7=471, 12=521, 17=509, 2=489, 13=506, 9=549, 6=519, 
1=502, 14=477, 10=513, 5=503, 0=481} 
“Ii~ 


在 main0 中 ， 自 动 包装 机 制 将 随机 生成 的 int 转 换 为 HashMap 可 以 使 用 的 Integer 引 用 (不 能 
使 用 基本 类 型 的 容器 )。 如 果 键 不 在 容器 中 ，get0 方 法 将 返回 null (这 表示 该 数字 第 一 次 被 找到 ) 。 
否则 ，get0 方 法 将 产生 与 该 键 相 关联 的 Integer 值 ， 然 后 这 个 值 被 递增 (自动 包装 机 制 再 次 简化 
了 表达 式 ， 但 是 确实 发 生 了 对 Integer 的 包装 和 拆 包 ) 。 

下 面 的 示例 允许 你 使 用 一 个 String 描 述 来 查找 Pet， 它 还 展示 了 你 可 以 使 用 怎样 的 方法 通过 
使 用 containsKey0 和 containsValue0 来 测试 一 个 Map， 以 便 查看 它 是 否 包含 某 个 键 或 某 个 值 。 


/1/: hotding/PetHap.java 
import typeinfo.pets.*; 
import java.util 
import static net.mindview.util.Print.*; 








public class PetHap { 
public static void main(String{] args) { 
Map<String,Pet> petMap = new HashMap<String,Pet>() ; 
petMap.put("My Cat", new Cat("Molly")); 
petMap.put ("My Dog", new Dog("Ginger")); 
petMap.put("My Hamster", new Hamster ("Bosco")); 
print (petMap) ; 
Pet dog = petMap.get("My Dog"): 
print (dog) ; 
print (petMap.containskey("My Dog")); 
print (petHap. containsValue (dog)) ; 
} 
} /* Output: 
{My Cat=Cat Molly, My Hamster=Hamster Bosco, My Dog=Dog 
Ginger} 
Dog Ginger 
true 
true 
AAA :~ 


Map 与 数组 和 其 他 的 Collection 一 样 ， 可 以 很 容易 地 扩展 到 多 维 ， 而 我 们 只 需 将 其 值 设 置 为 
Map (这 些 Map 的 值 可 以 是 其 他 容器 ， 甚 至 是 其 他 Map)。 因 此 ， 我 们 能 够 很 容易 地 将 容器 组 合 
起 来 从 而 快速 地 生成 强大 的 数据 结构 。 例 如 ， 假 设 你 正在 跟踪 拥有 多 个 宠物 的 人 ， 你 所 需 只 是 
一 个 Map<Person, List<Pet>>: 


/1/: holding/MapOfList. java 
package holding; 
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import typeinfo.pets.*; 
import java.util.*; 
import static net.mindview.util.Print 





public class MapOfList { 
public static Map<Person, List<? extends Pet>> 
petPeople = new HashMap<Person, List<? extends Pet>>(); 
static { 
petPeople.put (new Person("Dawn"), 
Arrays.asList(new Cymric("Ħolly"),new Mutt("Spot"))); 
petPeople.put(new Person("Kate"), 
Arrays.asList(new Cat("Shackleton"), 
new Cat("Elsie May"), new Dog("Margrett"))); 
petPeople.put(new Person("Marilyn"), 
Arrays .asList( 
new Pug("Louie aka Louis Snorkelstein Dupree"), 
new Cat("Stanford aka Stinky el Negro"), 
new Cat("Pinkola"))); 
petPeopte. put (new Person("Luke"), 
Arrays.asList(new Rat("Fuzzy"), new Rat("Fizzy"))); 
petPeople. put (new Person("Isaac"), 
Arrays.asList(new Rat("Freckly"))); 
} 
public static void main(String[] args) { 
print("People: ”+ petPeople.keySet()); 
print("Pets: ”+ petPeople.values()); 
for(Person person : petPeople.keySet()) { 
print(person + " has:"); 
for(Pet pet : petPeople.get(person)) 
print(" ”+ pet); 
} 
} 
} /* Output: 
People: [Person Luke, Person Marilyn, Person Isaac, Person 
Dawn, Person Kate] 
Pets: ({Rat Fuzzy, Rat Fizzy], [Pug Louie aka Louis 
Snorkelstein Dupree, Cat Stanford aka Stinky el Negro, Cat 
Pinkola}, [Rat Freckly], [Cymric Molly, Mutt Spot], (Cat 
Shackleton, Cat Elsie May, Dog Margrett)) 
Person Luke has: 
Rat Fuzzy 
Rat Fizzy 
Person Marilyn has: 
Pug Loufe aka Louis Snorkelstein Dupree 421 
Cat Stanford aka Stinky el Negro 
Cat Pinkola 
Person Isaac has: 
Rat Freckly 
Person Dawn has: 
Cymric Molly 
Mutt Spot 
Person Kate has: 
Cat Shackleton 
Cat Elsie May 
Dog Margrett 
Whim 


Map 可 以 返回 它 的 键 的 Set， 它 的 值 的 Collection， 或 者 它 的 键 值 对 的 Set。keySet0 方 法 产生 
了 由 在 petPeople 中 的 所 有 键 组 成 的 Set， 它 在 foreach 语 句 中 被 用 来 迭代 遍历 该 Map。 

练习 17: (2) 使 用 练习 1 中 的 Gerbil 类 ， 将 其 放 和 Map 中， 将 每 个 Gerbil 的 名 字 (例如 Fuzzy 
或 Spot) String (f) 与 每 个 Gerbil ( 值 ) 关联 起 来 。 为 keySet0 获 取 Iterator， 使 用 它 遍历 Map， 
针对 每 个 “ 键 ”查询 Gerbil， 然 后 打印 出 “ 键 "， 并 让 gerbil 执 行 hop0。 

练习 18: (3) 用 键 值 对 填充 一 个 HashMap。 打 印 结果 ， 通 过 散 列 码 来 展示 其 排序 。 抽 取 这 些 
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键 值 对 ， 按 照 键 进行 排序 ， 并 将 结果 置 于 一 个 LinkedHashMap 中 。 展 示 其 所 维护 的 插入 顺序 。 

练习 19: (2) 使 用 HashSet 和 LinkedHashSet 重 复 前 一 个 练习 。 

练习 20: (3) 修改 练习 16， 使 得 你 可 以 跟踪 每 一 个 元 音字 母 出 现 的 次 数 。 

练习 21: (3) 通过 使 用 Map<String.Integer>, 遵循 UniqueWordsjava 的 形式 来 创建 一 个 程序 ， 
它 可 以 对 一 个 文件 中 出 现 的 单词 计数 。 使 用 带 有 第 二 个 参数 String.CASE_INSENSITIVE_ 
ORDER 的 Collections.sort(0 方 法 对 结果 进行 排序 (将 产生 字母 序 ) ， 然 后 显示 结果 。 

练习 22: (5) 修改 前 一 个 练习 ， 使 其 用 一 个 包含 有 一 个 String 域 和 一 个 计数 域 的 类 来 存储 每 
一 个 不 同 的 单词 ， 并 使 用 一 个 由 这 些 对 象 构成 的 Set 来 维护 单词 列表 。 

练习 23: (4) 从 Statisticsjava 开 始 ， 写 一 个 程序 ， 让 它 重复 做 测试 ， 观 察 是 否 某 个 数字 比 别 
的 数字 出 现 的 次 数 多 。 

练习 24: (2) 使 用 String“ 键 ”和 你 选择 的 对 象 填充 LinkedHashMap。 然 后 从 中 提取 键 值 对 ， 
以 键 排序 ， 然 后 重新 插入 此 Map。 

练习 25，(3) 创建 一 个 Map<String.ArrayList<Integer>>， 使 用 net.mindview.TextFile 来 打开 
一 个 文本 文件 ， 并 一 次 读 入 一 个 单词 (使 用 “WW+” 作 为 TextFile 构 造 器 的 第 二 个 参数 )。 在 读 
人 单词 时 对 它们 进行 计数 ， 并 且 对 于 文件 中 的 每 一 个 单词 ， 都 在 ArrayList<Integer> 中 记录 下 与 
这 个 词 相 关联 的 单词 计数 。 实 际 上 ， 它 记录 的 是 该 单词 在 文件 中 被 发 现 的 位 置 。 

练习 26，(4) 拿 到 前 一 个 练习 中 所 产生 的 Map， 并 按照 它们 在 最 初 的 文件 中 出 现 的 顺序 重新 
创建 单词 顺序 。 


11.11 Queue 





队列 是 一 个 典型 的 先进 先 出 〈FIFO) 的 容器 。 即 从 容器 的 一 端 放 入 事物 ， 从 另 一 端 取出 ， 
并 且 事 物 放 入 容器 的 顺序 与 取出 的 顺序 是 相同 的 。 队 列 常 被 当 作 一 种 可 靠 的 将 对 象 从 程序 的 某 
个 区 域 传输 到 另 一 个 区 域 的 途径 。 队 列 在 并 发 编程 中 特别 重要 ， 就 像 你 将 在 第 21 章 中 所 看 到 的 ， 
因为 它们 可 以 安全 地 将 对 象 从 一 个 任务 传输 给 另 一 个 任务 。 。 

LinkedList 提 供 了 方法 以 支持 队列 的 行为 ， 并 且 它 实现 了 Queue 接 口 ， 因 此 LinkedList 可 以 
用 作 Queue 的 一 种 实现 。 通 过 将 LinkedList 向 上 转型 为 Queue， 下 面 的 示例 使 用 了 在 Queue 接 口 
中 与 Queue 相 关 的 方法 : 


//: holding/QueueDemo. java 
// Upcasting to a Queue from a LinkedList. 
import java.util.*; 


public class QueueDemo { 
public static void printQ(Queue queue) { 
while(queue.peek() != null) 
System. out .print (queue.remove() + * "); 
System.out .printin() ; 


} 
public static void main(String[] args) { 
Queue<Integer> queue = new LinkedList<Integer>(); 
Random rand = new Random(47); 
for(int i = 6; 1 < 10; i++) 
queue. offer(rand.nextInt(i + 10)): 
printQ(queue) ; 
Queue<Character> qc = new LinkedList<Character>(); 
for(char c : “Brontosaurus”.toCharArray()) 
qc.offer(c): r 
printQ(qc) ; 


} 
} /* Output: 
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offer0 方 法 是 与 Queue 相 关 的 方法 之 一 ， 它 在 允许 的 情况 下 ， 将 一 个 元 素 插入 到 队 尾 ， 或 者 
返回 false。peek0 和 element0 都 将 在 不 柳 除 的 情况 下 返回 队 头 但 是 peek0 方 法 在 队列 为 空 时 返 
回 null， 而 element0 会 抛 出 NoSuchElementException 异 常 。poll0 和 remove0 方 法 将 移 除 并 返回 
队 头 ， 但 是 poll0 在 队列 为 空 时 返回 noll， 而 remove0 会 抛 出 NoSuchElementException 异 常 。 

自动 包装 机 制 会 自动 地 将 nextInt0 方 法 的 int 结 果 转 换 为 queue 所 需 的 Integer 对 象 ， 将 char c 
转换 为 qe 所 需 的 Character 对 象 。Queue 接 口 窗 化 了 对 LinkedList 的 方法 的 访问 权限 ， 以 使 得 只 
有 恰当 的 方法 才 可 以 使 用 ， 因 此 ， 你 能 够 访问 的 LinkedList 的 方法 会 变 少 (这 里 你 实际 上 可 以 将 
queue 转 型 回 LinkedList， 但 是 至 少 我 们 不 鼓励 这 么 做 )。 

注意 ， 与 Queue 相 关 的 方法 提供 了 完整 而 独立 的 功能 。 即 ， 对 于 Queue 所 继承 的 Collection ， 
在 不 需要 使 用 它 的 任何 方法 的 情况 下 ， 就 可 以 拥有 一 个 可 用 的 Queue。 

练习 27: (2) 写 一 个 称 为 Command 的 类 ， 它 包含 一 个 String 域 和 一 个 显示 该 String 的 
operation0 方 法 。 写 第 二 个 类 ， 它 具有 一 个 使 用 Command 对 象 来 填充 一 个 Queue 并 返回 这 个 对 
象 的 方法 。 将 填充 后 的 Queue 传 递 给 第 三 个 类 的 一 个 方法 ， 该 方法 消耗 掉 Queue 中 的 对 象 ， 并 调 
用 它们 的 operation0 方 法 。 

11.11.1 PriorityQueue 

先进 先 出 描述 了 最 典型 的 队列 规则 。 队 列 规则 是 指 在 给 定 一 组 队列 中 的 元 素 的 情况 下 ， 确 
定 下 一 个 弹出 队列 的 元 素 的 规则 。 先 进 先 出 声明 的 是 下 一 个 元 素 应 该 是 等 待 时间 最 长 的 元 素 。 

优先 级 队列 声明 下 一 个 弹出 元 素 是 最 需要 的 元 素 (具有 最 高 的 优先 级 ) 。 例 如 ， 在 飞机 场 ， 
当 飞 机 临近 起 飞 时 ， 这 架 飞 机 的 乘客 可 以 在 办 理 登 机 手续 时 排 到 队 头 。 如 果 构 建 了 一 个 消息 系 
统 ， 某 些 消息 比 其 他 消息 更 重要 ， 因 而 应 该 更 快 地 得 到 处 理 ， 那 么 它们 何 时 得 到 处 理 就 与 它们 
何 时 到 达 无 关 。PriorityQueue 添 加 到 Java SE5 中 ， 是 为 了 提供 这 种 行为 的 一 种 自动 实现 。 

当 你 在 PriorityQueue 上 调用 offer0 方 法 来 插入 一 个 对 象 时 ， 这 个 对 象 会 在 队列 中 被 排序 。。 
默认 的 排序 将 使 用 对 象 在 队列 中 的 自然 顺序 ， 但 是 你 可 以 通过 提供 自己 的 Comparator 来 修改 这 
个 顺序 。PriorityQueue 可 以 确保 当 你 调用 peek0、poll0 和 remove0 方 法 时 ， 获 取 的 元 素 将 是 队 
列 中 优先 级 最 高 的 元 素 。 

让 PriorityQueue 与 Integer、String 和 Character 这 样 的 内 置 类 型 一 起 工作 易如反掌 。 在 下 面 
的 示例 中 ， 第 一 个 值 集 与 前 一 个 示例 中 的 随机 值 相 同 ， 因 此 你 可 以 看 到 它们 从 PriorityQueue 中 
弹出 的 顺序 与 前 一 个 示例 不 同 : 

//: holding/PriorityQueueDemo. java 

import java.util.*; 


public class PriorityQueueDemo { 
public static void main(String[] args) { 
PriorityQueue<Integer> priorityQueue = 
new PriorityQueue<Integer>(); 
Random rand = new Random(47) ; 
for(int i = 0; i < 10; i++) 
priorityQueue.offer(rand.nextint(i + 10)); 


QueueDemo. printQ(priorityQueue) ; 


O 这 实际 上 依赖 于 具体 实现 。 优 先 级 队列 算法 通常 会 在 插入 时 排序 (维护 一 个 堆 )， 但 是 它们 也 可 能 在 移 除 时 选 
择 最 重要 的 元 素 。 如 果 对 象 的 优先 级 在 它 在 队列 中 等 待 时 可 以 进行 修改 ， 那 么 算法 的 选择 就 显得 很 重要 了 。 
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List<Integer> ints = Arrays.asList(25, 22. 20, 
18, 14, 9, 3, 1, 1, 2, 3, 9, 14, 18, 21, 23, 25); 

priorityQueue = new PriorityQueue<Integer> (ints); 

Queuedemo. printQ(priorityQueue) ; 

PriorityQueue = new PriorityQueve<Integer>( 

ints.size(), Collections. reverseOrder()): 
priori tyQueue. addAll (ints); 
QueueDemo. printQ(priorityQueue) : 


String fact = "EDUCATION SHOULD ESCHEW OBFUSCATION™ ; 
List<String> strings = Arrays.asList(fact.split("")); 
PriorityQueue<String> stringPQ = 

new Priori tyQueue<String> (strings) ; 
QueueDemo. printQ(stringPQ); 
stringPQ = new PriorityQueue<String>( 

strings.size(), Collections. reverseOrder()); 
stringPQ.addAll (strings); 
QueueDemo. printQ(stringPQ); 


Set<Character> charSet = new HashSet<Character>(); 
for(char c : fact.toCharArray()) 

charSet.add(c); // Autoboxing 
PriorityQueue<Character> characterPQ = 

new PriorityQueue<Character>(charSet); 
QueueDemo. printQ(characterPQ) ; 


5 8 14 

14 14 18 18 20 21 22 23 25 25 

1 20 18 18 14 149933211 
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"ix 

你 可 以 看 到 ， 重 复 是 允许 的 ， 最 小 的 值 拥 有 最 高 的 优先 级 (如 果 是 String， 空 格 也 可 以 算 作 
值 ， 并 且 比 字母 的 优先 级 高 )。 为 了 展示 你 可 以 使 用 怎样 的 方法 通过 提供 自己 的 Comparator 对 
象 来 改变 排序 ， 第 三 个 对 PriorityQueue<Integer> 的 构造 器 调用 ， 和 第 二 个 对 PriorityQueue 
<String> 的 调用 使 用 了 由 Collection.reverseOrder() (新 添加 到 Java SE5 中 的 ) 产生 的 反 序 的 
Comparator, 

最 后 一 部 分 添加 了 一 个 HashSet 来 消除 重复 的 Character， 这 么 做 只 是 为 了 增添 点 乐趣 。 

Integer、String 和 Character 可 以 与 PriorityQueue 一 起 工作 ， 因 为 这 些 类 已 经 内 建 了 自然 
排序 。 如 果 你 想 在 PriorityQueue 中 使 用 自己 的 类 ， 就 必须 包括 额外 的 功能 以 产生 自然 排序 ， 或 
者 必须 提供 自己 的 Comparator。 在 第 17 章 中 有 一 个 更 加 复杂 的 示例 将 演示 这 种 情况 。 

练习 28: (2) 用 由 java.uti.Random 创 建 的 Double 值 填充 一 个 PriorityQueue (用 offer0 方 法 ) ， 
然后 使 用 poll0 移 除 并 显示 它们 。 

练习 29，(2) 创建 一 个 继承 自 Object 的 简单 类 ， 它 不 包含 任何 成 员 ， 展 示 你 不 能 将 这 个 类 的 
多 个 示例 成 功 地 添加 到 一 个 PriorityQueue 中 。 这 个 问题 将 在 第 17 章 中 详细 解释 。 


11.12 _ Collection 和 lterator 


Collection 是 描述 所 有 序列 容器 的 共性 的 根 接口 ， 它 可 能 会 被 认为 是 一 个 “附属 接口 "， 即 因 
为 要 表示 其 他 若干 个 接口 的 共性 而 出 现 的 接口 。 另 外 ，java.util.AbstractCollection 类 提供 了 
Collection 的 默认 实现 ， 使 得 你 可 以 创建 AbstractCollection 的 子 类 型 ， 而 其 中 没有 不 必要 的 代码 
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重复 。 

使 用 接口 描述 的 一 个 理由 是 它 可 以 使 我 们 能 够 创建 更 通用 的 代码 。 通 过 针对 接口 而 非 具 体 
实现 来 编写 代码 ， 我 们 的 代码 可 以 应 用 于 更 多 的 对 象 类 型 。 因 此 ， 如 果 我 编写 的 方法 将 接受 
一 个 Collection， 那 么 该 方法 就 可 以 应 用 于 任何 实现 了 Collection 的 类 一 一 这 也 就 使 得 一 个 新 类 
可 以 选择 去 实现 Collection 接 口 ， 以 便 我 的 方法 可 以 使 用 它 。 但 是 ， 有 一 点 很 有 趣 ， 就 是 我 们 
注意 到 标准 C++ 类 库 中 并 没有 其 容器 的 任何 公共 基 类 一 一 容器 之 间 的 所 有 共性 都 是 通过 迭代 器 
达成 的 。 在 Java 中 ， 遵 循 C++ 的 方式 看 起 来 似乎 很 明智 ， 即 用 和 迭代 器 而 不 是 Collection 来 表示 容 
器 之 间 的 共性 。 但 是 ， 这 两 种 方法 绑 定 到 了 一 起 ， 因 为 实现 Collection 就 意味 着 需要 提供 
iterator0 方 法 : 


//: holding/InterfaceVsIterator. java 
import typeinfo.pets.*; 
import java.util.*; 


public class InterfaceVsIterator { 
public static void display(Iterator<Pet> it) { 
while(it.nasNext()) { 
Pet p = it.next(); 
System.out.print(p.id() + ":" + p+" "); 


} 
System.out.printin(); 


} 
public static void display(Collection<Pet> pets) { 
for(Pet p : pets) 
System.out.print(p.id() + ":" +p +" "); 
System.out.printin(); 
$ 
public static void main(String[] args) { 
List<Pet> petList = Pets.arrayList (8); 
Set<Pet> petSet = new HashSet<Pet>(petList); 
Map<String,Pet> petMap = 
new LinkedHashMap<String,Pet>(): 
String(] names = ("Ralph, Eric, Robin, Lacey, " + 
"Britney, Sam, Spot, Fluffy").split(™. "); 
for(int 1 = 0; i < names.length; i++) 
petMap.put (names[i], petList.get(i)); 
display(petList); 
display (petSet); 
display(petList. iterator()); 
display(petSet. iterator()); 
System.out .printin(petHap); 
System.out.printin(petMap.keySet ()); 
display (petMap. values()); 
display (petMap.values() .iterator()): 


} 
} /* Output: 
@:Rat 1:Manx 2:Cymric 3:Mutt 4:Pug 5:Cymric 6:Pug 7:Manx 
4:Pug 6:Pug 3:Mutt 1:Manx 5:Cymric 7:Manx 2:Cymric @:Rat 
@:Rat 1:Manx 2:Cymric 3:Mutt 4:Pug S:Cymric 6:Pug 7:Manx 
4:Pug 6:Pug 3:Mutt 1:Manx 5:Cymric 7:Manx 2:Cymric @:Rat 
{Ralph=Rat, Eric=Manx, Robin=Cymric, Lacey=Mutt, 
Britney=Pug, Sam=Cymric, Spot=Pug, Fluffy=Manx} 
(Ralph, Eric, Robin, Lacey, Britney, Sam, Spot, Fluffy] 
@:Rat 1:Manx 2:Cymric 3:Mutt 4:Pug 5:Cymric 6:Pug 7:Manx 
@:Rat 1:Manx 2:Cymric 3:Mutt 4:Pug 5:Cymric 6:Pug 7:Manx 
Wha 


O 某 些 人 提倡 这 样 一 种 自动 创建 机 制 ， 即 对 一 个 类 中 所 有 可 能 的 方法 组 合 都 自动 创建 一 个 接口 ， 有 时 要 针对 每 
一 个 单个 的 类 都 自动 创建 。 我 相信 接口 的 意义 应 该 不 仅 限 于 方法 组 合 的 机 械 地 复制 ， 因 此 我 在 创建 接口 之 前 ， 
总 是 要 先 看 到 增加 接口 带 来 的 价值 。 
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两 个 版 本 的 display0 方 法 都 可 以 使 用 Map 或 Collection 的 子 类 型 来 工作 ， 而 且 Collection 接 口 
和 Iterator 都 可 以 将 display0 方 法 与 底层 容器 的 特定 实现 解 耦 。 

在 本 例 中 ， 这 两 种 方式 都 可 以 凌 效 。 事 实 上 ，Collection 要 更 方便 一 点 ， 因 为 它 是 Iterable 类 
型 ， 因 此 ， 在 display(Collection) 实 现 中 ， 可 以 使 用 foreach 结 构 ， 从 而 使 代码 更 加 清晰 。 

当 你 要 实现 一 个 不 是 Collection 的 外 部 类 时 ， 由 于 让 它 去 实现 Collection 接 口 可 能 非常 困难 或 
麻烦 ， 因 此 使 用 Iterator 就 会 变 得 非常 吸引 人 。 例 如 ， 如 果 我 们 通过 继承 一 个 持 有 Pet 对 象 的 类 
来 创建 一 个 Collection 的 实现 ， 那 么 我 们 必须 实现 所 有 的 Collection 方 法 ， 即 使 我 们 在 display0 方 
法 中 不 必 使 用 它们 ， 也 必须 如 此 。 尽 管 这 可 以 通过 继承 AbstractCollection 而 很 容易 地 实现 ， 但 
是 你 无 论 如 何 还 是 要 被 强制 去 实现 iterator0 和 size0 ， 以 便 提 供 AbstractCollection 没 有 实现 ， 但 
是 AbstractCollection 中 的 其 他 方法 会 使 用 到 的 方法 : 


//: holding/CollectionSequence. java 
import typeinfo.pets.*; 
import java.util.*: 


public class Collect ionSequence 
extends AbstractCollection<Pet> { 
private Pet[] pets = Pets.createArray(8) ; 
public int size() { return pets. length; } 
public Iterator<Pet> iterator() ( 
return new Iterator<Pet>() { 
private int index = 9; 
public boolean hasNext() { 
return index < pets. length; 


public Pet next() { return pets[index++]; } 
Public void remove() { // Not implemented 
throw new UnsupportedOperat ionException(); 


} 
oa 

} 

Public static void main(String() args) { 
CollectionSequence < = new CollectionSequence(); 
InterfaceVsIterator .display(c) ; 
InterfaceVsIterator .display(c.iterator()); 


} 
} /* Output: 
O:Rat 1:Manx 2:Cymric 3:Mutt 4:Pug 5:Cymric 6:Pug 7:Manx 
@:Rat 1:Manx 2:Cymric 3:Mutt 4:Pug 5:Cymric 6:Pug 7:Manx 
Whim 


remove( 方 法 是 一 个 “可 选 操作 ”， 你 将 在 第 17 章 中 了 解 它 。 这 里 不 必 实 现 它 ， 如 果 你 调用 
它 ， 它 会 抛 出 异常 。 

从 本 例 中 ， 你 可 以 看 到 ， 如 果 你 实现 Collection， 就 必须 实现 iterator()， 并 且 只 拿 实 现 
iterator0 与 继承 AbstractCollection 相 比 ， 花 费 的 代价 只 有 略微 减少 。 但 是 ， 如 果 你 的 类 已 经 继 
承 了 其 他 的 类 ， 那 么 你 就 不 能 再 继承 AbstractCollection 了 。 在 这 种 情况 下 ， 要 实现 Collection， 
就 必须 实现 该 接口 中 的 所 有 方法 。 此 时 ， 继 承 并 提供 创建 迭代 器 的 能 力 就 会 显得 容易 得 多 了 ， 


//: holding/NonCollectionSequence. java 
import typeinfo.pets.*; 
import java.util.*; 


class PetSequence { 
Protected Pet[] pets = Pets.createArray(8); 


public class NonCollectionSequence extends PetSequence { 
public Iterator<Pet> iterator() { 
return new Iterator<Pet>() { 
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private int index = 0; 
public boolean hasNext() { 


return index < pets.length; 





} 
public Pet next() { return pets{indext+] ; 
public void remove() { // Not implemented 
throw new UnsupportedOperat ionExcept ion() ; 
} 
}; 





public static void main(String{] args) { 
NonCollectionSequence nc = new NonCol lect ionSequence() ; 
InterfaceVsIterator .display(nc.iterator()): 


} 
} /* Output: 
@:Rat 1:Manx 2:Cymric 3:Mutt 4:Pug 5:Cymric 6:Pug 7:Manx 
“IIN: 


生成 Iterator 是 将 队列 与 消费 队列 的 方法 连接 在 一 起 耦合 度 最 小 的 方式 ， 并 且 与 实现 
Collection 相 比 ， 它 在 序列 类 上 所 施加 的 约束 也 少 得 多 。 

练习 30: (5) 修改 CollectionSequence.java， 使 其 不 要 继承 AbstractCollection， 而 是 实现 
Collection。 


11.13 ”Foreach 与 迭代 器 


到 目前 为 止 ，foreach 语 法 主要 用 于 数组 ， 但 是 它 也 可 以 应 用 于 任何 Collection 对 象 。 你 实际 
上 已 经 看 到 过 很 多 使 用 ArrayList 时 用 到 它 的 示例 ， 下 面 是 一 个 更 通用 的 证 明 : 


1/: hotding/ForEachCottections.java 
1/ ALL collections work with foreach. 
import java.util.*; 








public class For€achCollections { 
public static void main(String[] args) { 
Collection<String> cs = new LinkedList<String>(); 
Collections. addAll (cs, 
"Take the long way home".split(” ")); 
for (String 5 : cs) 
System.out.print(""" +s + "' "); 


} . Output: 

‘Take’ ‘the’ ‘long’ ‘way’ ‘home’ 

Wili~ 

由 于 cs 是 一 个 Collection， 所 以 这 段 代码 展示 了 能 够 与 foreach 一 起 工作 是 所 有 Collection 对 象 
的 特性 。 

之 所 以 能 够 工作 ， 是 因为 Java SE5 引 入 了 新 的 被 称 为 Iterable 的 接口 ， 该 接口 包含 一 个 能 够 
产生 Iterator 的 iterator0 方 法 ， 并 且 Iterable 接 口 被 foreach 用 来 在 序列 中 移动 。 因 此 如 果 你 创建 了 
任何 实现 Iterable 的 类 ， 都 可 以 将 它 用 于 foreach 语 名 中 : 

//: holding/IterableClass.java 


// Anything Iterable works with foreach. 
import java.util.*; 








public class IterableClass implements Iterable<String> { 
Protected String[] words = ("And that is how " + 
"we know the Earth to be banana-shaped.").split(" "); 
public Iterator<String> iterator() { 
return new Iterator<String>() { 
private int index = 6; 
public boolean hasNext() { 
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return index < words. length: 


public String next() { return words[index++]; } 
Public void remove() { // Not implemented 
throw new UnsupportedOperat ionExcept ion(): 
} 
N 
} 
; public static void main(String[] args) { 
for (String s : new IterableClass()) 
System.out.print(s + * "); 


$ 
} /* Output: 
And that is how we know the Earth to be banana-shaped. 
Whim 


iterator) HAE EMER T IteratorString> VE LAMA, YEAR ATLL 
历数 组 中 的 所 有 单词 。 在 main0 中 ， 你 可 以 看 到 IterableClass 确 实 可 以 用 于 foreach 语 句 中 。 

在 Java SE5 中 ， 大 量 的 类 都 是 Iterable 类 型 ， 主 要 包括 所 有 的 Collection 类 (但 是 不 包括 各 种 
Map)。 例 如 ， 下 面 的 代码 可 以 显示 所 有 的 操作 系统 环境 变量 : 


/1/: holding/EnvironmentVariables. java 
import java.util.*; 





public class EnvironmentVariables { 
public static void main(Stringl] args) ( 
1 for(Map.Entry entry: System.getenv() .entrySet()) ( 
System.out.printin(entry.getkey() +": ”+ 
entry.getValue()) ; 
} 


} A (Execute to see output) *///:~ 

System.getenv()° 返回 一 个 Map，entrySet0 产 生 一 个 由 Map.Entry 的 元 素 构成 的 Set， 并 且 
! 这 个 Set 是 一 个 Iterable， 因 此 它 可 以 用 于 foreach 循 环 。 

foreach 语 句 可 以 用 于 数组 或 其 他 任何 Iterable, 但 是 这 并 不 意味 着 数组 肯定 也 是 一 个 Iterable， 
' 而 任何 自动 包装 也 不 会 自动 发 生 ; 


//: holding/ArrayIsNotIterable. java 
import java.util.*; 


public class ArrayIsNotIterable { 
static <T> void test(Iterable<T> ib) { 
for(T t : ib) 
System.out.print(t +" "); 


} 
public static void sain(steingtl args) { 
test(Arrays.asList(1, 2, 3)) 
String[] strings = { a, "B", ce p 
// An array works in foreach, but it's not Iterable: 
/AL test (strings); 
// You must explicitly convert it to an Iterable: 
test (Arrays.asList(strings)); 


} 
| } /* Output: 


123ABC 
433, “H~ 


尝试 把 数组 当 作 一 个 Iterable 参 数 传递 会 导致 失败 。 这 说 明 不 存在 任何 从 数组 到 Iterable 的 

















© 在 Java SE5 之 前 还 没有 它 ， 因 为 该 方法 被 认为 与 操作 系统 的 境 合 度 过 紧 ， 
7 的 原则 。 现 在 提供 它 这 一 事实 表明 ，Java 的 设计 者 们 更 加 务实 了 。 





会 违反 “编写 一 次 ， 到 处 运行 
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自动 转换 ， 你 必须 手工 执行 这 种 转换 。 

练习 31: (3) 修改 polymorphism/shape/RandomShapeGeneratorjava， 使 其 成 为 一 个 Iterable。 
你 需要 添加 一 个 接收 元 素数 量 为 参数 的 构造 器 ， 这 个 数量 是 指 在 停止 之 前 ， 你 想 用 和 迭代 器 生成 
的 元 素 的 数量 。 验 证 这 个 程序 可 以 工作 。 
11.13.1 适配器 方法 惯用 法 

如 果 现 有 一 个 Iterable 类 ， 你 想 要 添加 一 种 或 多 种 在 foreach 语 句 中 使 用 这 个 类 的 方法 ， 应 该 
怎么 做 呢 ? 例如 ， 假 设 你 希望 可 以 选择 以 向 前 的 方向 或 是 向 后 的 方向 迭代 一 个 单词 列表 。 如 果 
直接 继承 这 个 类 ， 并 覆盖 iterator0 方 法 ， 你 只 能 替换 现 有 的 方法 ， 而 不 能 实现 选择 。 

一 种 解决 方案 是 所 谓 适配器 方法 的 惯用 法 。“ 适 配器 ”部 分 来 自 于 设计 模式 ， 因 为 你 必须 提 
供 特定 接口 以 满足 foreach 语 句 。 当 你 有 一 个 接口 并 需要 另 一 个 接口 时 ， 编 写 适 配器 就 可 以 解决 
问题 。 这 里 ， 我 希望 在 默认 的 前 向 迭代 器 的 基础 上 ， 添 加 产生 反 向 迭代 器 的 能 力 ， 因 此 我 不 能 
使 用 覆盖 ， 而 是 添加 了 一 个 能 够 产生 Iterable 对 象 的 方法 ， 该 对 象 可 以 用 于 foreach 语 句 。 正 如 你 
所 见 ， 这 使 得 我 们 可 以 提供 多 种 使 用 foreach 的 方式 : 


//: holding/AdapterMethodIdiom. java 

1/ The "Adapter Method" idiom allows you to use foreach 
// with additional kinds of Iterables. 

import java.util.*; 


class Reversiblearraylist<T> extends ArrayList<T> ( 
public ReversibleArrayList(Collection<T> c) { super(c); } 
public Iterable<T> reversed() { 
return new Iterable<T>() { 
public Iterator<T> iterator() { 
return new Iterator<T>() { 
int current = size() - 1; 
public boolean hasNext() { return current > -1; } 
public T next() { return get(current--); } 
public void remove() { // Not implemented 
throw new UnsupportedOperationExcept ton(); 


public class AdapterMethodIdiom { 
public static void main(String[] args) { 
ReversibleArrayList<String> ral = 
new ReversibleArrayList<String>( 
Arrays.asList("To be or not to be".split(" "))): 
// Grabs the ordinary iterator via iterator(): 
for (String s : ral) 
System.out.print(s +" "); 
System.out.printin(): 
/1/ Wand it the Iterable of your choice 
for (String s : ral.reversed()) 
System.out.print(s +" "); 


} 
} /* Output: 
To be or not to be 
be to not or be To 
“Whim 


如 果 直 接 将 ral 对 象 置 于 foreach 语 句 中 ， 将 得 到 (默认 的 ) 前 向 迭代 器 。 但 是 如 果 在 该 对 象 
上 调用 reversed( 方 法 ， 就 会 产生 不 同 的 行为 。 
通过 使 用 这 种 方式 ， 我 可 以 在 IterableClassjava 示 例 中 添加 两 种 适配器 方法 : 


























436} 
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/1/: holding/MultilterableClass.java 
// Adding several Adapter Methods. 
import java.util 





public class MultilterableClass extends IterableClass { 
public Iterable<String> reversed() { 
return new Iterable<String>() { 
public Iterator<String> iterator() { 

return new Iterator<String>() { 
int current = words. length ~ 
public boolean hasNext() { return current > -1; } 
public String next() { return words{current--]; } 

public void remove() { // Not implemented 

throw new UnsupportedOperat ionExcept ion() ; 








public Iterable<String> randomized() { 
return new Iterable<String>() { 
public Iterator<String> iterator() { 
List<String> shuffled = 
new ArrayList<String>(Arrays.asList(words)) ; 
Collections. shuffle(shuffled, new Random(47)); 
return shuffled. iterator(); 
} 





} 
public static void main(String{] args) { 
MultiIterableClass mic = new MultiIterableClass(); 
for (String s : mic.reversed()) 
System.out.print(s + * "); 
system. out.printin(); 
for (String s : mic.randomized()) 
System.out.print(s +" "); 
System.out.printin(); 
for(String s : mic) 
System.out.print(s +" "); 
} 
} /* Output: 
banana-shaped. be to Earth the know we how is that And 
is banana-shaped. Earth that how the be And we know to 
And that is how we know the Earth to be banana-shaped. 
“Ii~ 


注意 ， 第 二 个 方法 randomO 没 有 创建 它 自己 的 Iterator， 而 是 直接 返回 被 打 乱 的 List 中 的 


Iterator, 


从 输出 中 可 以 看 到 ，Collection.shuffle0 方 法 没有 影响 到 原来 的 数组 ， 而 只 是 打 乱 了 shuffled 


中 的 引用 。 之 所 以 这 样 ， 只 是 因为 randomized0 方 法 用 一 个 ArrayList 将 Arrays.asList0 方 法 的 结 
果 包 装 了 起 来 。 如 果 这 个 由 Arrays.asList0 方 法 产生 的 List 被 直接 打 乱 ， 那 么 它 就 会 修改 底层 的 
数组 ， 就 像 下 面 这 样 





//: holding/Modi fyingArraysAsList. java 

import java.util.*; 

public class ModifyingArraysAsList { 

public static void main(String{] args) { 
Random rand = new Random(47); 
Integer(] ia = { 1, 2, 3, 4. 5, 6. 7, 8, 9, 10}: 
List<Integer> listi = 
new ArrayList<Integer>(Arrays.asList(ia)); 

System.out.printin("Before shuffling: ”+ list): 
Collections. shuffle(listi, rand): 
System.out.printin("After shuffling: " + list1); 
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System.out.printin("array: ”+ Arrays. toString(ia)): 


List<Integer> list2 = Arrays.asList(ia): 

System. out.printin("Before shuffling: ”+ list2): 
Collections. shuffle(list2, rand); 
System.out.printin("After shuffling: ”+ list2); 
System.out.printin("array: " + Arrays. toString(ia)): 


} 
) /* Output: 
Before shuffling: [1, 2, 3. 4, 5. 6, 7, 8, 9, 10] 
After shuffling: (4, 6, 3, 1, 8. 7, 2, 5, 10, 9] 
array: {1, 2, 3, 4, 5, 6, 7, 8, 9, 10) 
Before shuffling: (1, 2. 3, 4, 5, 6. 7, 8, 9, 10) 
After shuffling: (9, 1. 6, 3. 7, 2, 5, 10, 4, 8] 
array: [9, 1, 6. 3, 7, 2, 5, 10, 4, 8] 
Wh 


在 第 一 种 情况 中 ，Arrays.asList0 的 输出 被 传递 给 了 ArrayList0 的 构造 器 ， 这 将 创建 一 个 引 
用 ia 的 元 素 的 ArrayList， 因 此 打 乱 这 些 引用 不 会 修改 该 数组 。 但 是 ， 如 果 直 接 使 用 
Arrays.asList(ia) 的 结果 ， 这 种 打 乱 就 会 修改 ia 的 顺序 。 意 识 到 Arrays.asList0 产 生 的 List 对 象 会 
使 用 底层 数组 作为 其 物理 实现 是 很 重要 的 。 只 要 你 执行 的 操作 会 修改 这 个 List， 并 且 你 不 想 原来 
的 数组 被 修改 ， 那 么 你 就 应 该 在 另 一 个 容器 中 创建 一 个 副本 。 

练习 32: (2) 按照 MultiIterableClass 示 例 ， 在 NonCollectionSequence.java 中 添加 reversed0 
和 randomized0 方 法 ， 并 让 NonCollectionSequence 实 现 Iterable。 然 后 在 foreach 诸 句 中 展示 所 有 
的 使 用 方式 。 


11.14 总 结 


Java 提 供 了 大 量 持 有 对 象 的 方式 : 

1) 数组 将 数字 与 对 象 联系 起 来 。 它 保存 类 型 明确 的 对 象 ， 查 询 对 象 时 ， 不 需要 对 结果 做 
类 型 转换 。 它 可 以 是 多 维 的 ， 可 以 保存 基本 类 型 的 数据 。 但 是 ， 数 组 一 旦 生成 ， 其 容量 就 不 
能 改变 。 

2) Collection 保 存单 一 的 元 素 ， 而 Map 保 存 相关 联 的 键 值 对 。 有 了 Java 的 泛 型 ， 你 就 可 以 指 
定 容器 中 存放 的 对 象 类 型 ， 因 此 你 就 不 会 将 错误 类 型 的 对 象 放 置 到 容器 中 ， 并 且 在 从 容器 中 获 
取 元 素 时 ， 不 必 进 行 类 型 转换 。 各 种 Collection 和 各 种 Map 都 可 以 在 你 向 其 中 添加 更 多 的 元 素 时 ， 
自动 调整 其 尺寸 。 容 器 不 能 桂 有 基本 类 型 ， 但 是 自动 包装 机 制 会 仔细 地 执行 基本 类 型 到 容器 中 
所 持 有 的 包装 器 类 型 之 间 的 双向 转换 。 

3) 像 数 组 一 样 ，List 也 建立 数字 索引 与 对 象 的 关联 ， 因 此 ， 数 组 和 List 都 是 排 好 序 的 容器 。 
List 能 够 自动 扩充 容量 。 

4) 如 果 要 进行 大 量 的 随机 访问 ， 就 使 用 ArrayList， 如 果 要 经 常 从 表 中 间 插 入 或 删除 元 素 ， 
则 应 该 使 用 LinkedList。 

5) 各 种 Queue 以 及 栈 的 行为 ， 由 LinkedList 提 供 支持 。 

© Map 是 一 种 将 对 象 (而 非 数字 ) 与 对 象 相关 联 的 设计 。HashMap 设 计 用 来 快速 访问 ， 而 
TreeMap 保 持 “ 键 ”始终 处 于 排序 状态 ， 所 以 没有 HashMap 快 。LinkedHashMap 保 持 元 素 插入 
的 顺序 ， 但 是 也 通过 散 列 提供 了 快速 访问 能 力 。 

7) Set 不 接受 重复 元 素 。HashSet 提 供 最 快 的 查询 速度 ， 而 TreeSet 保 持 元 素 处 于 排序 状态 。 
LinkedHashSet 以 插入 顺序 保存 元 素 。 

8) 新 程序 中 不 应 该 使 用 过 时 的 Vector、Hashtable 和 Stack。 

浏览 一 下 Java 容 器 的 简 图 (不 包含 抽象 类 和 遗留 构件 ) 会 大 有 神 益 。 这 里 只 包含 你 在 一 一 般 
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情况 下 会 磁 到 的 接口 和 类 。 





{ Collection 一 
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LinkedHashSet 
简单 的 容器 分 类 


你 可 以 看 到 ， 其 实 只 有 四 种 容器 : Map、List、Set 和 Queue， 它 们 各 有 两 到 三 个 实现 版 本 
(Queue 的 java.utilconcurrent 实 现 没有 包括 在 上 面 这 张 图 中 )。 常 用 的 容器 用 黑色 粗 线 框 表示 。 

点 线 框 表示 接口 ， 实 线 框 表示 普通 的 (具体 的 ) 类 。 带 有 空心 箭头 的 点 线 表 示 一 个 特定 的 
类 实现 了 一 个 接口 ， 实 心 箭头 表示 某 个 类 可 以 生成 箭头 所 指向 类 的 对 象 。 例 如 ， 任 意 的 
Collection 可 以 生成 Iterator， 而 List 可 以 生成 ListIterator (也 能 生成 普通 的 Iterator， 因 为 List 继 
Jk É Collection) , j 

下 面 的 示例 展示 了 各 种 不 同 的 类 在 方法 上 的 差异 。 实 际 的 代码 来 自 第 15 章 ， 我 在 这 里 只 是 
调用 它 以 产生 输出 。 程 序 的 输出 也 展示 了 在 每 个 类 或 接口 中 所 实现 的 接口 ; 


/1/: holding/ContainerHethods .java 
import net.mindview.util.*; 








public class ContainerMethods { 
public static void main(String{] args) { 
ContainerMethodDi fferences.main(args); 


$ 

} /* Output: (Sample) 
Collection: [add, addAll, clear, contains, containsAll, 
equals, hashCode, isEmpty, iterator, remove, removeAll, 

“ retainAll, size, toArray] 
Interfaces in Collection: [Iterable] 
Set extends Collection, adds: [] 
Interfaces in Set: [Collection] 
HashSet extends Set, adds: [] 
Interfaces in HashSet: [Set, Cloneable, Serializable] 
LinkedHashSet extends HashSet, adds: [] 
Interfaces in LinkedHashSet: [Set, Cloneable, Serializable] 
TreeSet extends Set, adds: [pollLast, navigableHeadset, 
descendingIterator, lower, headSet, ceiling, pollFirst, 
subSet, navigableTailset, comparator, first, floor, last, 
navigableSubSet, higher, tailSet] 
Interfaces in TreeSet: [NavigableSet. Cloneable, 
Serializable] 
List extends Collection, adds: [listIterator, indexof, get, 
subList, set. lastIndexOf] 
Interfaces in List: [Collection] 
ArrayList extends List, adds: [ensureCapacity, trimToSize] 
Interfaces in ArrayList: [List, RandomAccess, Cloneable, 
Serializable) 
LinkedList extends List, adds: [pollLast, offer, 
descendinglterator, addFirst, peekLast, removeFirst, 
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peekFirst, removelast, getLast, pollFirst, pop, poll, 
addLast, removeFirstOccurrence, getFirst, element, peek, 
offerLast, push, offerFirst, removeLastOccurrence 
Interfaces in LinkedList: [List, Deque, Cloneable 
Serializable] 

Queue extends Collection, adds: [offer, element, peek, 
poll] 

“Interfaces in Queue: [Collection] 

PriorityQueue extends Queue, adds: [comparator 
Interfaces in PriorityQueue: [Serializable] 

Map: [clear, containsKey, containsValue, entrySet, equals, 
get, hashCode, isEmpty, keySet, put, putAll, remove, size 
values] 

HashMap extends Map, adds: [] 

Interfaces in HashMap: [Map, Cloneable, Serializable] 
LinkedHashMap extends HashMap, adds: [] 

Interfaces in LinkedHashMap: [Map] 

SortedMap extends Hap, adds: [subMap, comparator, firstKey, 
lastKey, headMap, tailMap] 

Interfaces in SortedMap: [Map] 

TreeMap extends Map, adds: [descendingEntrySet, subMap, 
pollLastEntry, lastkey, floorEntry, lastEntry, lowerKey 
navigableHeadMap, navigableTailMap, descendingkeySet, 
tailMap, ceilingEntry, higherKey, pollFirstentry, 
comparator, firstKey, floorkey, higherEntry, firstEntry 
navigableSubMap, headMap, lowerEntry, ceilingkey. 
Interfaces in TreeMap: (NavigableMap, Cloneable, 
Serializable] 

Wh 


可 以 看 到 ， 除 了 TreeSet 之 外 的 所 有 Set 都 拥有 与 Collection 完 全 一 样 的 接口 。List 和 
Collection 存 在 着 明显 的 不 同 ， 尽 管 List 所 要 求 的 方法 都 在 Collection 中 。 另 一 方面 ， 在 Queue 接 
口中 的 方法 都 是 独立 的 ， 在 创建 具有 Queue 功 能 的 实现 时 ， 不 需要 使 用 Colleetion 方 法 。 最 后 ， 
Map 和 Collection 之 间 的 唯一 重合 就 是 Map 可 以 使 用 entrySetO 和 values0 方 法 来 产生 Collection。 

注意 ， 标 记 接口 java.uti.RandomAccess 附 着 到 了 ArrayList 上 ， 而 没有 附着 到 LinkedList 上 。 
这 为 想 要 根据 所 使 用 的 特定 的 List 而 动态 修改 其 行为 的 算法 提供 了 信息 。 

从 面向 对 象 的 继承 层次 关系 来 看 ， 这 种 组 织 结构 确实 有 些 奇 怪 。 但 是 ， 当 你 了 解 了 java.util 
中 更 多 的 有 关 容 器 的 内 容 后 (特别 是 第 17 章 中 的 内 容 ) ， 你 就 会 看 到 除了 继承 结构 有 些 奇怪 外 ， 
还 有 更 多 的 问题 。 容 器 类 库 一 直 以 来 都 是 设计 难题 一 -解决 这 些 难题 涉及 到 要 去 满足 经 常 彼此 
之 间 互 为 牵制 的 各 方面 需求 。 因 此 你 应 该 学 会 中 庸 之 道 。 

抛 开 这 些 问 题 ，Java 的 容器 每 天 都 会 用 到 的 工具 ， 它 可 以 使 程序 更 简洁 、 更 强大 、 更 高 效 。 
在 适应 容器 类 库 的 某 些 方面 之 前 ， 你 确实 得 废 点 劲 儿 ， 但 是 我 想 你 很 快 就 会 找到 自己 的 路 子 ， 
去 获得 和 使 用 这 个 类 库 中 的 类 。 

所 选 习题 的 答案 都 可 以 在 名 为 The Thinking in Java Annotated Solution Guide 的 电子 文档 中 找 
到 ， 读 者 可 以 从 www.MindView.net 购 买 此 文档 。 








第 12 章 通过 异常 处 理 错误 


Java 的 基本 理念 是 “结构 不 佳 的 代码 不 能 运行 "。 

发 现 错误 的 理想 时 机 是 在 编译 阶段 ， 也 就 是 在 你 试图 运行 程序 之 前 。 然 而 ， 编 译 期 间 并 不 
能 找 出 所 有 的 错误 ， 余下 的 问题 必须 在 运行 期 间 解 决 。 这 就 需要 错误 源 能 通过 某 种 方式 ， 把 适 
当 的 信息 传递 给 某 个 接收 者 一 该 接收 者 将 知道 如 何 正确 处 理 这 个 问题 。 

改进 的 错误 恢复 机 制 是 提供 代码 健壮 性 的 最 强 有 力 的 方式 。 错 误 恢复 在 我 们 所 编写 的 每 一 
个 程序 中 都 是 基本 的 要 素 ， 但 是 在 Java 中 它 显 得 格外 重要 ， 因 为 Java 的 主要 目标 之 一 就 是 创建 供 
他 人 使 用 的 程序 构件 。 要 想 创建 健壮 的 系统 ， 它 的 每 一 个 构件 都 必须 是 健壮 的 。Java 使 用 异常 
来 提供 一 致 的 错误 报告 模型 ， 使 得 构件 能 够 与 客户 端 代 码 可 靠 地 沟通 问题 。 

Java 中 的 异常 处 理 的 目的 在 于 通过 使 用 少 于 目前 数量 的 代码 来 简化 大 型 、 可 靠 的 程序 的 生 
成 ， 并 且 通 过 这 种 方式 可 以 使 你 更 加 自信 : 你 的 应 用 中 没有 未 处 理 的 错误 。 异 常 的 相关 知识 学 
起 来 并 非 艰 涩 难 懂 ， 并 且 它 属于 那 种 可 以 使 你 的 项 目 受益 明显 、 立 竿 见 影 的 特性 之 一 。 

因为 异常 处 理 是 Java 中 唯一 正式 的 错误 报告 机 制 ， 并 且 通 过 编译 器 强制 执行 ， 所 以 不 学 习 
异常 处 理 的 话 ， 书 中 也 就 只 能 写 出 那么 些 例子 了 。 本 章 将 向 读者 介绍 如 何 编写 正确 的 异常 处 理 
程序 ， 并 将 展示 当 方 法 出 问题 的 时 候 ， 如 何 产生 自 定义 的 异常 。 


12.1 概念 


C 以 及 其 他 早期 语言 常常 具有 多 种 错误 处 理 模式 , 这 些 模式 往往 建立 在 约定 俗 成 的 基础 之 上 
而 并 不 属于 语言 的 一 部 分 。 通 常会 返回 某 个 特殊 值 或 者 设置 某 个 标志 ， 并 且 假定 接收 者 将 对 这 
个 返回 值 或 标志 进行 检查 ， 以 判定 是 否 发 生 了 错误 。 然 而 ， 随 着 时 间 的 推移 ， 人 们 发 现 ， 高 做 
的 程序 员 们 在 使 用 程序 库 的 时 候 更 倾向 于 认为 :“ 对 ， 错 误 也 许 会 发 生 ， 但 那 是 别人 造成 的 ， 不 
关 我 的 事 ”"。 所 以 ， 程 序 员 不 去 检查 错误 情形 也 就 不 足 为 奇 了 何况 对 某 些 错误 情形 的 检查 确实 
很 无 聊 )。 如 果 的 确 在 每 次 调用 方法 的 时 候 都 彻底 地 进行 错误 检查 ， 代 码 很 可 能 会 变 得 难以 阅 
读 。 正 是 由 于 程序 员 还 仍然 用 这 些 方式 拼凑 系统 ， 所 以 他 们 拒绝 承认 这 样 一 个 事实 : 对 于 构造 
大 型 、 健 壮 、 可 维护 的 程序 而 言 ， 这 种 错误 处 理 模式 已 经 成 为 了 主要 障碍 。 

解决 的 办 法 是 ， 用 强制 规定 的 形式 来 消除 错误 处 理 过程 中 随心 所 欲 的 因素 。 这 种 做 法 由 来 
已 久 ， 对 异常 处 理 的 实现 可 以 追溯 到 20 世 纪 60 年 代 的 操作 系统 ， 甚 至 于 BASIC 语 言 中 的 on error 
goto 语 句 。 而 C++ 的 异常 处 理 机 制 基 于 Ada，Java 中 的 异常 处 理 则 建立 在 C++ 的 基础 之 上 (尽管 
看 上 去 更 像 Object Pascal) 。 

“异常 ”这 个 词 有 “我 对 此 感到 意外 ”的 意思 。 问 题 出 现 了 ， 你 也 许 不 清楚 该 如 何 处 理 ， 但 
你 的 确 知道 不 应 该 置之不理 ， 你 要 停 下 来 ， 看 看 是 不 是 有 别人 或 在 别 的 地 方 ， 能 够 处 理 这 个 问 
题 。 只 是 在 当前 的 环境 中 还 没有 足够 的 信息 来 解决 这 个 问题 ， 所 以 就 把 这 个 问题 提交 到 一 个 更 
高 级 别 的 环境 中 ， 在 这 里 将 作出 正确 的 决定 。 

使 用 异常 所 带 来 的 另 一 个 相当 明显 的 好 处 是 ， 它 往往 能 够 降低 错误 处 理 代 码 的 复杂 度 。 如 


O 比如 ，C 程 序 员 检 查 printf0 的 返回 值 就 是 这 样 。 
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果 不 使 用 异常 ， 那 么 就 必须 检查 特定 的 错误 ， 并 在 程序 中 的 许多 地 方 去 处 理 它 。 而 如 果 使 用 异 
常 ， 那 就 不 必 在 方法 调用 处 进行 检查 ， 因 为 异常 机 制 将 保证 能 够 捕获 这 个 错误 。 H, REE 

一 个 地 方 处 理 错误 ， 即 所 谓 的 异常 处 理 程序 中 。 这 种 方式 不 仅 节省 代码 ， 而 且 把 “描述 在 正常 
执行 过 程 中 做 什么 事 ” 的 代码 和 “出 了 问题 怎么 办 ”的 代码 相 分 离 。 总 之 ， 与 以 前 的 错误 处 理 
方法 相 比 ， 异 常 机制 使 代码 的 阅读 、 编 写 和 调试 工作 更 加 井井有条 。 


12.2 基本 异常 


异常 情形 (exceptional condition) 是 指 阻止 当前 方法 或 作用 域 继续 执行 的 问题 。 把 异常 情形 
与 普通 问题 相 区 分 很 重要 ， 所 谓 的 普通 问题 是 指 ， 在 当前 环境 下 能 得 到 足够 的 信息 ， 总 能 处 理 
这 个 错误 。 而 对 于 异常 情形 ， 就 不 能 继续 下 去 了 ， 因 为 在 当前 环境 下 无 法 获得 必要 的 信息 来 解 
决 问题 。 你 所 能 做 的 就 是 从 当前 环境 跳出 ， 并 且 把 问题 提交 给 上 一 级 环境 。 这 就 是 抛 出 异常 时 
所 发 生 的 事情 

除法 就 是 一 个 简单 的 例子 。 除 数 有 可 能 为 0， 所 以 先进 行 检查 很 有 必要 。 但 除数 为 0 代表 的 
究竟 是 什么 意思 呢 ? 通过 当前 正在 解决 的 问题 环境 ， 或 许 能 知道 该 如 何 处 理 除数 为 0 的 情况 。 但 
如 果 这 是 一 个 意料 之 外 的 值 ， 你 也 不 清楚 该 如 何 处 理 ， 那 就 要 抛 出 异常 ， 而 不 是 顺 着 原来 的 路 
径 继续 执行 下 去 。 

当 抛 出 异常 后 ， 有 几 件 事 会 随 之 发 生 。 首 先 ， 同 Java 中 其 他 对 象 的 创建 一 样 ， 将 使 用 new 在 
堆 上 创建 异常 对 象 。 然 后 ， 当 前 的 执行 路 径 〈 它 不 能 继续 下 去 了 ) 被 终止 ， 并 且 从 当前 环境 中 
弹出 对 异常 对 象 的 引用 。 此 时 ， 异 常 处 理 机 制 接管 程序 ， 并 开始 寻找 一 个 恰当 的 地 方 来 继续 执 
行程 序 。 这 个 恰当 的 地 方 就 是 异常 处 理 程序 ， 它 的 任务 是 将 程序 从 错误 状态 中 恢复 ， 以 使 程序 
能 要 么 换 一 种 方式 运行 ， 要 么 继续 运行 下 去 。 

举 一 个 抛 出 异常 的 简单 例子 。 对 于 对 象 引用 t， 传 给 你 的 时 候 可 能 尚未 被 初始 化 。 所 以 在 使 
用 这 个 对 象 引用 调用 其 方法 之 前 ， 会 先 对 引用 进行 检查 。 可 以 创建 一 个 代表 错误 信息 的 对 象 
并 且 将 它 从 当前 环境 中 “ 抛 出 "， 这 样 就 把 错误 信息 传播 到 了 “更 大 ”的 环境 中 。 这 被 称 为 抛 出 
一 个 异常 ， 看 起 来 像 这 样 : [445 


1f(t == null) 
throw new NullPointerException(); 














这 就 抛 出 了 异常 ， 于 是 在 当前 环境 下 就 不 必 再 为 这 个 问题 操心 了 ， 它 将 在 别 的 地 方 得 到 处 理 。 
具体 是 哪个 “地 方 ”后 面 很 快 就 会 介绍 。 

异常 使 得 我 们 可 以 将 每 件 事 都 当 作 一 个 事务 来 考虑 ， 而 异常 可 以 看 护 着 这 些 事务 的 底 
线 “…… 事 务 的 基本 保障 是 我 们 所 需 的 在 分 布 式 计算 中 的 异常 处 理 。 事 务 是 计算 机 中 的 合同 法 ， 
如 果 出 了 什么 问题 ， 我 们 只 需要 放弃 整个 计算 。”。 我 们 还 可 以 将 异常 看 作 是 一 种 内 建 的 恢复 
(undo) 系 统 ， 因 为 (在 细心 使 用 的 情况 下 ) 我 们 在 程序 中 可 以 拥有 各 种 不 同 的 恢复 点 。 如 果 程序 
的 某 部 分 失败 了 ， 异 常 将 “恢复 ”到 程序 中 某 个 已 知 的 稳定 点 上 。 

异常 最 重要 的 方面 之 一 就 是 如 果 发 生 问题 ， 它 们 将 不 允许 程序 沿 着 其 正常 的 路 径 继续 走 下 
去 。 在 C 和 C++ 这 样 的 语言 中 ， 这 可 真是 个 问题 ， 尤 其 是 C， 它 没有 任何 办 法 可 以 强制 程序 在 出 
现 问题 时 停止 在 某 条 路 径 上 运行 下 去 ， 因 此 我 们 有 可 能 会 较 长 时 间 地 忽略 了 问题 ， 从 而 陷入 了 
完全 不 恰当 的 状态 中 。 蜡 常 允 许 我 们 (如果 没有 其 他 手段 ) 强制 程序 停止 运行 ， 并 告诉 我 们 出 
现 了 什么 问题 , 或 者 (理想 状态 下 ) 强制 程序 处 理 问题 ， 并 返回 到 稳定 状态 。 


© Jim Gray，www.acmqueue.org 的 一 次 访谈 中 提 到 ， 由 于 他 的 团队 在 事务 方面 的 杰出 贡献 而 成 为 图 灵 奖 得 主 。 
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12.21 异常 参数 

与 使 用 Java 中 的 其 他 对 象 一 样 ， 我 们 总 是 用 new 在 堆 上 创建 异常 对 象 ， 这 也 伴随 着 存储 空间 
的 分 配 和 构造 器 的 调用 。 所 有 标准 异常 类 都 有 两 个 构造 器 : 一 个 是 默认 构造 器 ， 另 一 个 是 接受 
字符 串 作为 参数 ， 以 便 能 把 相关 信息 放 人 异常 对 象 的 构造 器 

throw new NullPointerException("t = null"): 

不 久 读者 将 看 到 ， 要 把 这 个 字符 串 的 内 容 提取 出 来 可 以 有 多 种 不 同 的 方法 。 

关键 字 throw 将 产生 许多 有 趣 的 结果 。 在 使 用 new 创 建 了 异常 对 象 之 后 ， 此 对 象 的 引用 将 传 
给 throw。 尽 管 返 回 的 异常 对 象 其 类 型 通常 与 方法 设计 的 返回 类 型 不 同 ， 但 从 效果 上 看 ， 它 就 像 
是 从 方法 “返回 ”的 。 可 以 简单 地 把 异常 处 理 看 成 一 种 不 同 的 返回 机 制 ， 当 然 若 过 分 强调 这 种 
类 比 的 话 ， 就 会 有 麻烦 了 。 另 外 还 能 用 抛 出 异常 的 方式 从 当前 的 作用 域 退 出 。 在 这 两 种 情况 下 
将 会 返回 一 个 异常 对 象 ， 然 后 退出 方法 或 作用 域 。 

抛 出 异常 与 方法 正常 返回 值 的 相似 之 处 到 此 为 止 。 因 为 异常 返回 的 “地 点 ”与 普通 方法 调 
用 返回 的 “地 点 ”完全 不 同 。( 异 常 将 在 一 个 恰当 的 异常 处 理 程序 中 得 到 解决 ， 它 的 位 置 可 能 离 
异常 被 抛 出 的 地 方 很 远 ， 也 可 能 会 跨越 方法 调用 栈 的 许多 层次 。) 

此 外 ， 能 够 抛 出 任意 类 型 的 Throwable 对 象 ， 它 是 异常 类 型 的 根 类 。 通 常 ， 对 于 不 同类 型 的 
错误 ， 要 抛 出 相应 的 异常 。 错 误 信息 可 以 保存 在 异常 对 象 内 部 或 者 用 异常 类 的 名 称 来 暗示 。 上 
一 层 环境 通过 这 些 信息 来 决定 如 何 处 理 异 常 。( 通 常 ， 异 常 对 象 中 仅 有 的 信息 就 是 异常 类 型 ， 除 
此 之 外 不 包含 任何 有 意义 的 内 容 。) 


12.3 捕获 异常 


要 明白 异常 是 如 何 被 捕获 的 ， 必 须 首先 理解 监控 区 域 (guarded region) 的 概念 。 它 是 一 段 
可 能 产生 异常 的 代码 ， 并 且 后 面 跟着 处 理 这 些 异常 的 代码 。 


12.3.1 tyik 

如 果 在 方法 内 部 抛 出 了 异常 (或 者 在 方法 内 部 调用 的 其 他 方法 抛 出 了 异常 )， 这 个 方法 将 在 
抛 出 异常 的 过 程 中 结束 。 要 是 不 希望 方法 就 此 结束 ， 可 以 在 方法 内 设置 一 个 特殊 的 块 来 捕获 异 
常 。 因 为 在 这 个 块 里 “尝试 ”各 种 (可 能 产生 异常 的 ) 方法 调用 ， 所 以 称 为 ty 块 。 它 是 跟 在 try 
关键 字 之 后 的 普通 程序 块 ， 


try { 
1/ Code that might generate exceptions 
} 


对 于 不 支持 异常 处 理 的 程序 语言 ， 要 想 仔细 检查 错误 ， 就 得 在 每 个 方法 调用 的 前 后 加 上 设 
置 和 错误 检查 的 代码 ， 甚 至 在 每 次 调用 同一 方法 时 也 得 这 么 做 。 有 了 异常 处 理 机 制 ， 可 以 把 所 
有 动作 都 放 在 try 块 里 ， 然 后 只 需 在 一 个 地 方 就 可 以 捕获 所 有 异常 。 这 意味 着 代码 将 更 容易 编写 
和 阅读 ， 因 为 完成 任务 的 代码 没有 与 错误 检查 的 代码 混在 一 起 。 

12.3.2 异常 处 理 程序 

当然 ， 抛 出 的 异常 必须 在 某 处 得 到 处 理 。 这 个 “地 点 ”就 是 异常 处 理 程序 ， 而 且 针对 每 个 要 
捕获 的 异常 ， 得 准备 相应 的 处 理 程序 。 异 常 处 理 程序 紧 跟 在 ty 块 之 后 ， 以 关键 字 catch 表 示 

oie that might generate exceptions 

} catch(Typel idl) { 


// Handle exceptions of Typel 
} catch(Type2 id2) { 
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// Handle exceptions of Type2 
} catch(Type3 id3) { 

// Handle exceptions of Type3 
} 


II etc... 

每 个 catch 子 名 (异常 处 理 程序 ) 看 起 来 就 像 是 接收 一 个 且 仅 接收 一 个 特殊 类 型 的 参数 的 方 
法 。 可 以 在 处 理 程序 的 内 部 使 用 标识 符 (id1，id2 等 等 )， 这 与 方法 参数 的 使 用 很 相似 。 有 时 可 
能 用 不 到 标识 符 ， 因 为 异常 的 类 型 已 经 给 了 你 足够 的 信息 来 对 异常 进行 处 理 ， 但 标识 符 并 不 可 
以 省 略 。 

异常 处 理 程序 必须 紧 跟 在 try 块 之 后 。 当 异常 被 抛 出 时 ， 异 常 处 理 机 制 将 负责 搜寻 参数 与 异 
常 类 型 相 匹配 的 第 一 个 处 理 程序 。 然 后 进入 catch 子 句 执行 ， 此 时 认为 异常 得 到 了 处 理 。 一 旦 
catch 子 句 结束 ， 则 处 理 程 序 的 查找 过 程 结束 。 注 意 ， 只 有 匹配 的 catch 子 句 才 能 得 到 执行 ， 这 与 
Switch 语句 不 同 ，switch 语 句 需要 在 每 一 个 case 后 面 跟 一 个 break ， 以 避免 执行 后 续 的 case 子 句 。 

注意 在 try 块 的 内 部 ， 许 多 不 同 的 方法 调用 可 能 会 产生 类 型 相同 的 异常 ， 而 你 只 需要 提供 一 
个 针对 此 类 型 的 异常 处 理 程序 。 

终止 与 恢复 

异常 处 理 理论 上 有 两 种 基本 模型 。Java 支 持 终止 模型 ( 它 是 Java 和 C++ 所 支持 的 模型 ) ©, 
在 这 种 模型 中 ， 将 假设 错误 非常 关键 ， 以 至 于 程序 无 法 返回 到 异常 发 生 的 地 方 继续 执行 。 一 旦 
异常 被 抛 出 ， 就 表明 错误 已 无 法 挽回 ， 也 不 能 回来 继续 执行 。 

另 一 种 称 为 恢复 模型 。 意 思 是 异常 处 理 程序 的 工作 是 修正 错误 ， 然 后 重新 尝试 调用 出 问题 
的 方法 ， 并 认为 第 二 次 能 成 功 。 对 于 恢复 模型 ， 通 常 希望 异常 被 处 理 之 后 能 继续 执行 程序 。 如 
果 想 要 用 Java 实 现 类 似 恢复 的 行为 ， 那 么 在 遇见 错误 时 就 不 能 抛 出 异常 ， 而 是 调用 方法 来 修正 
该 错误 。 或 者 ， 把 try 块 放 在 while 循 环 里 ， 这 样 就 不 断 地 进入 try 块 ， 直 到 得 到 满意 的 结果 。 

长 久 以 来 ， 尽 管 程序 员 们 使 用 的 操作 系统 支持 恢复 模型 的 异常 处 理 ， 但 他 们 最 终 还 是 转向 
使 用 类 似 “ 终 止 模型 ”的 代码 ， 并 且 忽 略 恢复 行为 。 所 以 虽然 恢复 模型 开始 显得 很 吸引 人 ， 但 
不 是 很 实用 。 其 中 的 主要 原因 可 能 是 它 所 导致 的 耦合 : 恢复 性 的 处 理 程序 需要 了 解 异 常 好 出 的 
地 点 ， 这 势必 要 包含 依赖 于 抛 出 位 置 的 非 通用 性 代码 。 这 增加 了 代码 编写 和 维护 的 困难 ， 对 于 
异常 可 能 会 从 许多 地 方 抛 出 的 大 型 程序 来 说 ， 更 是 如 此 。 


124 创建 自 定义 异常 


不 必 拘 泥 于 Java 中 已 有 的 异常 类 型 。Java 提 供 的 异常 体系 不 可 能 预见 所 有 的 希望 加 以 报告 的 
错误 ， 所 以 可 以 自己 定义 异常 类 来 表示 程序 中 可 能 会 遇 到 的 特定 问题 。 

要 自己 定义 异常 类 ， 必 须 从 已 有 的 异常 类 继承 ， 最 好 是 选择 意思 相近 的 异常 类 继承 (不 过 
这 样 的 异常 并 不 容易 找 )。 建 立新 的 异常 类 型 最 简单 的 方法 就 是 让 编译 器 为 你 产生 默认 构造 器 ， 
所 以 这 几乎 不 用 写 多 少 代码 : 


//: exceptions/InheritingExceptions. java 
// Creating your own exceptions. 


class SimpleException extends Exception {} 
public class InheritingExceptions { 
public void f() throws SimpleException { 


日 这 与 大 多 数 语言 的 机 制 相同 ， 包 括 C++、C#、Python 和 D 等 语言 。 
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System.out.println("Throw SimpleException from f()"); 
throw new SimpleException(); 


} 
public static void main(String{] args) { 
InheritingExceptions sed = new InheritingExceptions(); 
try { 
sed.f0; 
} catch(SimpleException e) { 
System.out.printin("Caught it!"): 
} 


} 
} /* Output: 
Throw SimpleException from f() 
Caught it! 
Whim 


编译 器 创建 了 默认 构造 器 ， 它 将 自动 调用 基 类 的 默认 构造 器 。 本 例 中 不 会 得 到 像 Simple- 
Exception(String) 这 样 的 构造 器 ， 这 种 构造 器 也 不 实用 。 你 将 看 到 ， 对 异常 来 说 ， 最 重要 的 部 
分 就 是 类 名 ， 所 以 本 例 中 建立 的 异常 类 在 大 多 数 情况 下 已 经 够 用 了 。 

本 例 的 结果 被 打印 到 了 控制 台 上 ， 本 书 的 输出 显示 系统 正 是 在 控制 台 上 自动 地 捕获 和 测试 
这 些 结果 的 。 但 是 ， 你 也 许 想 通过 写 人 System.err 而 将 错误 发 送 给 标准 错误 流 。 通 常 这 比 把 错误 
信息 输出 到 System.out 要 好 ， 因 为 System.out 也 许 会 被 重 定向 。 如 果 把 结果 送 到 System.err， 它 

就 不 会 随 System.out 一 起 被 重 定 向 ， 这 样 更 容易 被 用 户 注意 。 
也 可 以 为 异常 类 定义 一 个 接受 字符 串 参数 的 构造 器 ; 


/1/: exceptions/FullConstructors.java 


class MyException extends Exception { 

public MyException() {} 

public MyException(String msg) { super (msg) 
} 


public class FullConstructors { 
public static void f() throws MyException { 
System.out.printin("Throwing MyException from f()"); 
throw new MyException(); 





} 

public static void g() throws MyException { 
System. out.printin("Throwing MyException from g()"); 
throw new MyException("Originated in g()"); 


} 
public static void main(String[] args) { 


} catch(MyException e) { 
e.printStackTrace(System.out) ; 
$ 
try { 
g0; 
} catch(MyException e) { 
e.printStackTrace(System.out); 
} 
} 
} /* Output: 
Throwing MyException from f() 
‘HyException 
at FullConstructors.f(FullConstructors.java:11) 
at FullConstructors.main(FullConstructors.java:19) 
Throwing MyException from g() 
MyException: Originated in g() 
at FuliConstructors.g(FullConstructors.java:15) 
at FullConstructors.main(FullConstructors.java:24) 
Wh 
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新 增 的 代码 不 长 : 两 个 构造 器 定义 了 MyException 类 型 对 象 的 创建 方式 。 对 于 第 二 个 构造 器 ， 
使 用 super 关 键 字 明确 调用 了 其 基 类 构造 器 ， 它 接受 一 个 字符 串 作为 参数 。 

在 异常 处 理 程序 中 ， 调 用 了 在 Throwable 类 声明 (Exception 即 从 此 类 继承 ) 的 
PrintStackTrace0 方 法 。 就 像 从 输出 中 看 到 的 ， 它 将 打印 “从 方法 调用 处 直到 异常 抛 出 处 ”的 方 
法 调用 序列 。 这 里 ， 信 息 被 发 送 到 了 System.out， 并 自动 地 被 捕获 和 显示 在 输出 中 。 但 是 ， 如 果 
调用 默认 版 本 ; 

e.printStackTrace(); 

则 信息 将 被 输出 到 标准 错误 流 。 

练习 1， (2) 编写 一 个 类 ， 在 其 main0 方 法 的 try 块 里 抛 出 一 个 Exception 类 的 对 象 。 传 递 一 个 
字符 串 参 数 给 Exception 的 构造 器 。 在 catch 子 句 里 捕获 此 异常 对 象 ， 并 且 打 印字 符 串 参数 。 添 加 
一 个 finally 子 句 ， 打 印 一 条 信息 以 证 明 这 里 确实 得 到 了 执行 。 

练习 2: (1) 定义 一 个 对 象 引用 并 初始 化 为 noll， 尝 试用 此 引用 调用 方法 。 把 这 个 调用 放 在 
try-catch 子 句 里 以 捕获 异常 。 

练习 3，(1) 编写 能 产生 并 能 捕获 ArrayIndexOutOfBoundsException 异 常 的 代码 。 

练习 4:(2) 使 用 extends 关 键 字 建立 一 个 自 定义 异常 类 。 为 这 个 类 写 一 个 接受 字符 申 参数 的 
构造 器 ， 把 此 参数 保存 在 对 象 内 部 的 字符 串 引 用 中 。 写 一 个 方法 显示 此 字符 串 。 写 一 个 try- 
catch 子 句 ， 对 这 个 新 异常 进行 测试 。 

练习 5: (3) 使 用 while 循 环 建立 类 似 “ 恢 复 模型 ”的 异常 处 理 行为 ， 它 将 不 断 重 复 ， 直 到 异 
常 不 再 抛 出 。 

12.4.1 异常 与 记录 日 志 

你 可 能 还 想 使 用 javautil.logging 工 具 将 输出 记录 到 日 志 中 。 尽 管 记录 日 志 的 全 部 细节 是 在 
http://MindView.net/Books/BetterJava 的 补充 材料 中 介绍 的 ， 但 是 这 里 所 使 用 的 基本 的 日 志 记 录 功 
能 还 是 相当 简单 易 慌 的 ， 


//: exceptions/LoggingExceptions.java 
/1/ An exception that reports through a Logger. 

import java.util. logging.*; 

import java.io.*; 


class LoggingException extends Exception { 
private static Logger logger = 
Logger. getLogger ("LoggingException"); 
public LoggingException() { 
StringWriter trace = new Stringhriter(); 
printStackTrace(new PrintWriter(trace)); 
logger. severe(trace.toString()): 


$ 


public class LoggingExceptions { 
public static void main(String{) args) { 
try { 
throw new LoggingException() ; 
} catch(LoggingException e) { 
System.err.printin("Caught * + e); 
} 
try { 
throw new LoggingException(): 
} catch(LoggingException e) { 
System.err.printin("Caught * + e); 
} 


} 
} /* Output: (85% match) 
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Aug 30, 2005 4:02:31 PM LoggingException 
SEVERE: LoggingExcept ion 
at 


LoggingExcept ions .main(LoggingExceptions. 


Caught Loggingexception 
Aug 30, 2005 4:02:31 PM LoggingException 
SEVERE: LoggingException 

at 


LoggingExceptions.main(LoggingExceptions. 


Caught Loggingexception 
Wh 


<init> 


java:19) 


<init> 


java:24) 


静态 的 LoggergetLogger( 方 法 创建 了 一 个 String 参 数 相关 联 的 Logger 对 象 (通常 与 错误 相 


053 关 的 包 名 和 类 名 )， 这 个 Logger 对 象 会 将 其 输出 发 送 到 System.err。 向 Logger 写 入 的 最 简单 方式 


就 是 直接 调用 与 日 志 记录 消息 的 级 别 相关 联 的 方法 ， 这 里 使 用 的 是 severe0。 为 了 产生 日 志 记录 
消息 ， 我 们 欲 获取 异常 抽出 处 的 栈 轨迹 ， 但 是 printStackTrace0 不 会 默认 地 产生 字符 种 。 为 了 获 
取 字符 串 ， 我 们 需要 使 用 重 载 的 printStackTrace( 方 法 ， 它 接受 一 个 java.io.PrintWriter 对 象 作 
为 参数 (这 些 都 将 在 第 18 章 中 详细 解释 )。 如 果 我 们 将 一 个 java.io.StringWriter 对 象 传递 给 这 个 
PrintWriter 的 构造 器 ， 那 么 通过 调用 toString0 方 法 ， 就 可 以 将 输出 抽取 为 一 个 String。 

尽管 由 于 LoggingException 将 所 有 记录 日 志 的 基础 设施 都 构建 在 异常 自身 中 ， 使 得 它 所 使 
用 的 方式 非常 方便 ， 并 因此 不 需要 客户 端 程序 员 的 干预 就 可 以 自动 运行 ， 但 是 更 常见 的 情形 是 
我 们 需要 捕获 和 记录 其 他 人 编写 的 异常 ， 因 此 我 们 必须 在 异常 处 理 程序 中 生成 日 志 消息 : 


//: exceptions/LoggingExceptions2. java 
// Logging caught exceptions, 

import java.util. logging. *; 

import java.io.*; 


public class LoggingExceptions2 { 
Private static Logger logger = 


Logger . getLogger ("LoggingExceptions2"); 


Static void logException(Exception e) { 


StringWriter trace = new Stringwriter(); 
e.printStackTrace(new Printhriter(trace)) ; 


logger .severe(trace.toString()); 
} 


public static void main(String{] args) { 


try { 
throw new NullPointerException() ; 
} catch(NullPointerException e) { 
logexception(e) ; 


) 
} /* Output: (98% match) 


Aug 30, 2005 4:07:54 PM LoggingExceptions2 logException 


SEVERE: java. lang.NullPointerException 
at 


LoggingExceptions2.main(LoggingExceptions2. java: 16) 


Wim 


还 可 以 更 进一步 自 定义 异常 ， 比 如 加 入 额外 的 构造 器 和 成 员 : 


//: exceptions/ExtraFeatures. java 


// Further embellishment of exception classes. 


import static net.mindview.util.Print.*; 


class MyException2 extends Exception { 
private int x; 
public MyException2() {} 


public MyException2(String msg) { super(msg); } 
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public MyException2(String msg, int x) { 





public int val() { return x; } 
public String getMessage() { 
return "Detail Message: "+ x + " "+ super.getMessage(); 
} 
} 


public class ExtraFeatures { 

public static void f() throws MyException2 { 
print("Throwing MyException2 from f()"); 
throw new MyException2(); 

$ 

public static void g() throws MyException2 { 
print("Throwing MyException2 from g()"); 
throw new MyException2 ("Originated in g()"): 

t 

public static void h() throws MyException2 { 
print("Throwing MyException2 from h()"); 
throw new MyException2("Originated in h()", 47); 


} 
public static void main(String[] args) { 
try { 
tO; 
} catch(MyException2 e) { 
e.printStackTrace(System.out) ; 
$ 
try { 


g0: 
} catch(MyException2 e) { 
e.printStackTrace(System.out) ; 
} 
try { 
hO; 
} catch(MyException2 e) { 
e.printStackTrace (System. out); 
System.out.printin("e.val() = ”+ e.val()); 
} 


} 

} /* Output: 

Throwing MyException2 from f() 

MyException2: Detail Message: @ null 
at ExtraFeatures. f(ExtraFeatures. java:22) 
at ExtraFeatures.main(ExtraFeatures. java:34) 

Throwing HyException2 from g() 

MyException2: Detail Message: @ Originated in g() 
at ExtraFeatures.g(ExtraFeatures. java:26) 
at ExtraFeatures.main(ExtraFeatures. java:39) 

Throwing MyException2 from h() 

MyException2: Detail Message: 47 Originated in hO 
at ExtraFeatures.n(ExtraFeatures. java: 30) 
at ExtraFeatures.main(ExtraFeatures. java:44) 

e.val() = 47 

Whim 


新 的 异常 添加 了 字段 x 以 及 设 定 x 值 的 构造 器 和 读 取 数据 的 方法 。 此 外 ， 还 覆盖 了 Throwable. 
getMessage() 方 法 ， 以 产生 更 详细 的 信息 。 对 于 异常 类 来 说 ，getMessage() 方 法 有 点 类 似 于 
toString0 方 法 。 

既然 异常 也 是 对 象 的 一 种 ， 所 以 可 以 继续 修改 这 个 异常 类 ， 以 得 到 更 强 的 功能 。 但 要 记 住 ， 
使 用 程序 包 的 客户 端 程序 员 可 能 仅仅 只 是 查看 一 下 抛 出 的 异常 类 型 ， 其 他 的 就 不 管 了 (大 多 数 
Java 库 里 的 异常 都 是 这 么 用 的 ) ， 所 以 对 异常 所 添加 的 其 他 功能 也 许 根本 用 不 上 。 

练习 6: (1) 创建 两 个 异常 类 ， 每 一 个 都 自动 记录 它们 自己 的 日 志 ， 演 示 它 们 都 可 以 正常 运行 。 
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练习 7: (D 修改 练习 3， 使 得 catch 子 名 可 以 将 结果 作为 日 志 记 录 。 
12.5 异常 说 明 


Java 鼓 励 人 们 把 方法 可 能 会 抛 出 的 异常 告知 使 用 此 方法 的 客户 端 程序 员 。 这 是 种 优雅 的 做 
法 ， 它 使 得 调用 者 能 确切 知道 写 什 么 样 的 代码 可 以 捕获 所 有 潜在 的 异常 。 当 然 ， 如 果 提 供 了 源 
代码 ， 客 户 端 程序 员 可 以 在 源 代码 中 查找 throw 语 句 来 获知 相关 信息 ， 然 而 程序 库 通常 并 不 与 源 
代码 一 起 发 布 。 为 了 预防 这 样 的 问题 ，Java 提 供 了 相应 的 语法 (并 强制 使 用 这 个 语法 )， 使 你 能 
以 礼 驳 的 方式 告知 客户 端 程序 员 某 个 方法 可 能 会 抛 出 的 异常 类 型 ， 然 后 客户 端 程序 员 就 可 以 进 
行 相应 的 处 理 。 这 就 是 异常 说 明 ， 它 属于 方法 声明 的 一 部 分 ， 紧 跟 在 形式 参数 列表 之 后 。 

异常 说 明 使 用 了 附加 的 关键 字 throws， 后 面 接 一 个 所 有 潜在 异常 类 型 的 列表 ， 所 以 方法 定 
义 可 能 看 起 来 像 这 样 : 

但 是 ， 要 是 这 样 写 


void f() throws TooBig, TooSmall, DivZero { //... 


就 表示 此 方法 不 会 抛 出 任何 异常 (除了 从 RuntimeException 继 承 的 异常 ， 它 们 可 以 在 没有 异常 

void tO { /1/ ... 
说 明 的 情况 下 被 抛 出 ， 这 些 将 在 后 面 进行 讨论 )。 

代码 必须 与 异常 说 明 保持 一 致 。 如 果 方 法 里 的 代码 产生 了 异常 却 没有 进行 处 理 ， 编 译 器 会 
发 现 这 个 问题 并 提醒 你 :要么 处 理 这 个 异常 ， 要 么 就 在 异常 说 明 中 表明 此 方法 将 产生 异常 。 通 
过 这 种 自 顶 向 下 强制 执行 的 异常 说 明 机 制 ，Java 在 编译 时 就 可 以 保证 一 定 水 平 的 异常 正确 性 。 

ABA MME “PEM” ED: 可 以 声明 方法 将 抛 出 异常 ， 实 际 上 却 不 抛 出 。 编 译 器 相 
信 了 这 个 声明 ， 并 强制 此 方法 的 用 户 像 真 的 抛 出 异常 那样 使 用 这 个 方法 。 这 样 做 的 好 处 是 ， 为 
异常 先 占 个 位 子 ， 以 后 就 可 以 抛 出 这 种 异常 而 不 用 修改 已 有 的 代码 。 在 定义 抽象 基 类 和 接口 时 
这 种 能 力 很 重要 ， 这 样 派生 类 或 接口 实现 就 能 够 抛 出 这 些 预先 声明 的 异常 。 

这 种 在 编译 时 被 强制 检查 的 异常 称 为 被 检查 的 异常 。 

练习 8: (1) 定义 一 个 类 ， 令 其 方法 抛 出 在 练习 2 里 定义 的 异常 。 不 用 异常 说 明 ， 看 看 能 否 通 
过 编译 。 然 后 加 上 异常 说 明 ， 用 try-catch 子 句 测试 该 类 和 异常 。 


12.6 捕获 所 有 异常 


可 以 只 写 一 个 异常 处 理 程序 来 捕获 所 有 类 型 的 异常 。 通 过 捕获 异常 类 型 的 基 类 Exception ， 
就 可 以 做 到 这 一 点 (事实 上 还 有 其 他 的 基 类 ， 但 Exception 是 同 编程 活动 相关 的 基 类 ): 


catch(Exception e) { 
System.out.printin(“Caught an exception"); 


这 将 捕获 所 有 异常 ， 所 以 最 好 把 它 放 在 处 理 程序 列表 的 未 尽 ， 以 防 它 抢 在 其 他 处 理 程序 之 前 先 
把 异常 捕获 了 。 

因为 Exception 是 与 编程 有 关 的 所 有 异常 类 的 基 类 ， 所 以 它 不 会 含有 太 多 具体 的 信息 ， 不 过 
可 以 调用 它 从 其 基 类 Throwable 继 承 的 方法 : 


String getMessage() 
String getLocalizedMessage() 


用 来 获取 详细 信息 ， 或 用 本 地 语言 表示 的 详细 信息 。 
String toString0 
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返回 对 Throwable 的 简单 描述 ， 要 是 有 详细 信息 的 话 ， 也 会 把 它 包 含 在 内 。 
void printStackTrace() 

void printStackTrace(PrintStream) 

void printStackTrace(java.io.PrintWriter) 
打印 Throwable 和 Throwable 的 调用 栈 轨迹 。 调 用 栈 显 示 了 -把 你 带 到 异常 抽出 地 点 的 方法 调 
用 序列 。 其 中 第 一 个 版 本 输出 到 标准 错误 ， 后 两 个 版 本 允许 选择 要 输出 的 济 (在 第 18 章 ， 你 将 
学 习 这 两 种 流 的 不 同 之 处 )。 

Throwable fillInStackTrace() 

用 于 在 Throwable 对 象 的 内 部 记录 栈 帧 的 当前 状态 。 这 在 程序 重新 抛 出 错误 或 异常 (很 快 就 会 讲 
到 ) 时 很 有 用 。 

此 外 ， 也 可 以 使 用 Throwable 从 其 基 类 Objeet (也 是 所 有 类 的 基 类 ) 继承 的 方法 。 对 于 异常 
来 说 ，getClass0 也 许 是 个 很 好 用 的 方法 ， 它 将 返回 一 个 表示 此 对 象 类 型 的 对 象 。 然 后 可 以 使 用 
getName() 方 法 查询 这 个 Class 对 象 包含 包 信息 的 名 称 ， 或 者 使 用 只 产生 类 名 称 的 getSimple 
Name0 方 法 。 

下 面 的 例子 演示 了 如 何 使 用 Exception 类 型 的 方法 : 


//: exceptions/ExceptionMethods . java 
// Demonstrating the Exception Methods. 
import static net.mindview.util.Print.*; 


public class ExceptionMethods { 

public static void main(String{] args) { 

try { 

throw new Exception("My Exception"); 
catch(Exception e) { 
print ("Caught Exception"); 
print ("getMessage():" + e.getMessage()); 
print ("getLocalizedMessage():" + 

e.getLocalizedMessage()); 
print("toString():" + e); 
print("printStackTrace():"); 
e.printStackTrace(System.out) ; 

} 
} 

} /* Output: 

Caught Except fon 

getMessage() :My Exception 
。 getLocalizedMessage() :My Exception 

toString():java.lang.Exception: My Exception 

printStackTrace(): 

java.lang.Exception: My Exception 

at ExceptionMethods .main(Except ionMethods.. java:8) 


Whim 


可 以 发 现 每 个 方法 都 比 前 一 个 提供 了 更 多 的 信息 一 一 实际 上 它们 每 一 个 都 是 前 一 个 的 超 集 。 
练习 9: (2) 定义 三 种 新 的 异常 类 型 。 写 一 个 类 ， 在 一 个 方法 中 抛 出 这 三 种 异常 。 在 main() 
里 调用 这 个 方法 ， 仅 用 一 个 cateh 子 句 捕获 这 三 种 异常 。 
12.6.1 栈 轨迹 
PrintStackTrace0 方 法 所 提供 的 信息 可 以 通过 getStackTrace0 方 法 来 直接 访问 ， 这 个 方法 将 
返回 一 个 由 栈 轨迹 中 的 元 素 所 构成 的 数组 ， 其 中 每 一 个 元 素 都 表示 栈 中 的 一 桢 。 元 素 0 是 栈 顶 元 
K, 并且 是 调用 序列 中 的 最 后 一 个 方法 调用 (这 个 Throwable 被 创建 和 抛 出 之 处 )。 数 组 中 的 最 
后 一 个 元 素 和 栈 底 是 调用 序列 中 的 第 一 个 方法 调用 。 下 面 的 程序 是 一 个 简单 的 演示 示例 : 
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//: exceptions/WhoCalled. java 
// Programmatic access to stack trace information. 


public class WhoCalled { 
static void f() ¢ 
// Generate an exception to fill in the stack trace 
try { 
throw new Exception(); 
} catch (Exception e) { 
for(StackTraceElement ste : e.getStackTrace()) 
System.out.println(ste.getMethodName()) ; 
} 


} 

static void g0 { fO; } 

static void h() { 80; } 

public static void main(Stringl] args) { 
tO; 


system.out.printtn(* 

80: 

System. out.printin(” 
0: 





} 
} /* Output: 
f 


这 里 ， 我 们 只 打印 了 方法 名 ， 但 实际 上 还 可 以 打印 整个 StackTraceElement， 它 包含 其 他 附 
件 的 信息 。 


12.6.2 重新 抛 出 异常 

有 时 希望 把 刚 捕 获 的 异常 重新 抛 出 ， 尤 其 是 在 使 用 Exception 捕 获 所 有 异常 的 时 候 。 既 然 已 
经 得 到 了 对 当前 异常 对 象 的 引用 ， 可 以 直接 把 它 重新 抛 出 : 

catch(Exception e) { 

System.out.printin("An exception was thrown"); 
throw e; 

} 

重 抛 异常 会 把 异常 抛 给 上 一 级 环境 中 的 异常 处 理 程序 ， 同 一 个 try 块 的 后 续 catch 子 句 将 被 忽 
略 。 此 外 ， 异 常 对 象 的 所 有 信息 都 得 以 保持 ， 所 以 高 一 级 环境 中 捕获 此 异常 的 处 理 程序 可 以 从 
这 个 异常 对 象 中 得 到 所 有 信息 。 

如 果 只 是 把 当前 异常 对 象 重新 抛 出 ， 那 么 printStackTrace0 方 法 显示 的 将 是 原来 异常 抛 出 点 
的 调用 栈 信息 ， 而 并 非 重新 抛 出 点 的 信息 。 要 想 更 新 这 个 信息 ， 可 以 调用 fihlInStackTrace( 方 法 ， 
这 将 返回 一 个 Throwable 对 象 ， 它 是 通过 把 当前 调用 栈 信息 填 人 原来 那个 异常 对 象 而 建立 的 ， 就 
像 这 样 : 


//: exceptions/Rethrowing. java 
// Demonstrating fillInStackTrace() + 


public class Rethrowing { 
public static void f() throws Exception { 
System.out.println("originating the exception in f()"); 
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throw new Exception("thrown from f()"); [461 
} 
Public static void g() throws Exception { 

try { = 
tO; 

} catch(Exception e) { 

System.out.printin("Inside g(),e.printStackTrace()"); 
e.printStackTrace(System.out) ; 
throw e; 
} 
} 
public static void h() throws Exception { 

try { 
fO; 

} catch(Exception e) { 

System.out.printin("Inside h(),e.printStackTrace()"); 
e.printStackTrace(System.out); 
throw (Exception)e. fillInStackTrace(); 
} 
} 
Public static void main(String(] args) { 
try { 
BO; 

} catch(Exception e) { 
System.out.printin("main: printStackTrace()"); 
e.printStackTrace(System.out): 

} 

try { 
hO; 

} catch(Exception e) { 
System.out.println("main: printStackTrace()"); 
e.printStackTrace(System.out) ; 

} 

} 
} /* Output: 
originating the exception in fO 
Inside g(),e.printStackTrace() 
java. lang.Exception: thrown from f() 
at Rethrowing. f (Rethrowing. java:7) 
at Rethrowing.g(Rethrowing. java: 11) 
at Rethrowing.main(Rethrowing. java:29) 
main: printStackTrace() 
java.lang.Exception: thrown from f() 
at Rethrowing. f (Rethrowing. java:7) 
at Rethrowing.g(Rethrowing. java: 11) 
at Rethrowing.main(Rethrowing. java:29) 
originating the exception in fO 
Inside h(),e.printStackTrace() 
java.lang.Exception: thrown from f() 
at Rethrowing.f (Rethrowing. java:7) 
at Rethrowing.h(Rethrowing.java:26) 
at Rethrowing.main(Rethrowing.java:35) 
main: printStackTrace() 
java.lang.Exception: thrown from f() 
at Rethrowing.h(Rethrowing. java:24) 
at Rethrowing.main(Rethrowing. java:35) 




















Wha 
调用 fillInStackTrace0 的 那 一 行 就 成 了 异常 的 新 发 生地 了 。 
有 可 能 在 捕获 异常 之 后 抛 出 另 一 种 异常 。 这么 做 的 话 , 得 到 的 效果 类 似 于 使 用 fillInStackTrace0， 
有 关 原 来 异常 发 生 点 的 信息 会 丢失 ， 剩 下 的 是 与 新 的 抛 出 点 有 关 的 信息 : 


//: exceptions/RethrowNew. java 
// Rethrow a different object from the one that was caught. 


class OneException extends Exception { 


[ee 
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public OneException(String s) { super(s); } 


class TwoException extends Exception { 
public TwoException(String s) { super(s): } 
} 


public class RethrowNew { 
public static void f() throws OneException { 

System.out.printin("originating the exception in f()"); 
throw new OneException(“thrown from f()"); 


public static void main(String{] args) { 
try { 


} catch(OneException e) { 
System.out .printtn( 
"Caught in inner try, e.printStackTrace()"); 
e.printStackTrace(System. out) ; 
throw new TwoException("from inner try"): 


} 
} catch(TwoException e) { 
System. out. printin( 
"Caught in outer try, e.printStackTrace()")s 
e.printStackTrace(System. out) ; 
) 
} 
i } /* Output: 
originating the exception in f() 
Caught in inner try, e.printStackTrace() 
OneException: thrown from f() 
1 at RethrowNew.f(RethrowNew.java:15) 
at RethrowNew .matn(RethrowNew.java:29) 
Caught in outer try, e.printStackTrace() 
1 TwoException: from inner try 
at RethrowNew.main(RethrowNew. java:25) 
t “Mi~ 


最 后 那个 异常 仅 知道 自己 来 自 main0， 而 对 f0 一 无 所 知 。 ` 

永远 不 必 为 清理 前 一 个 异常 对 象 而 担心 ， 或 者 说 为 异常 对 象 的 清理 而 担心 。 它 们 都 是 用 new 
在 堆 上 创建 的 对 象 ， 所 以 垃圾 回收 器 会 自动 把 它们 清理 掉 。 
12.6.3 异常 链 . 

常常 会 想 要 在 捕获 一 个 异常 后 抛 出 另 一 个 异常 ， 并 且 希 望 把 原始 异常 的 信息 保存 下 来 ， 这 
被 称 为 异常 链 。 在 JDK 1.4 以 前 ， 程 序 员 必 须 自己 编写 代码 来 保存 原始 异常 的 信息 。 现 在 所 有 
Throwable 的 子 类 在 构造 器 中 都 可 以 接受 一 个 cause (因由 ) 对 象 作为 参数 。 这 个 cause 就 用 来 表 
示 原 始 异常 这样 通 过 把 原始 异常 传递 给 新 的 异常 ， 使 得 即使 在 当前 位 置 创建 并 抛 出 了 新 的 异 
常 ， 也 能 通过 这 个 异常 链 追 踪 到 异常 最 初 发 生 的 位 置 。 

有 趣 的 是 ， 在 Throwable 的 子 类 中 ， 只 有 三 种 基本 的 异常 类 提供 了 带 cause 参 数 的 构造 器 。 
它们 是 Error (用 于 Java 虚 拟 机 报告 系统 错误 )、Exception 以 及 RuntimeException。 如 果 要 把 其 

[他 类 型 的 异常 链接 起 来 ， 应 该 使 用 initCause( 方 法 而 不 是 构造 器 。 
下 面 的 例子 能 让 你 在 运行 时 动态 地 向 DynamicFields 对 象 添加 字段 ， 


//: exceptions/DynamicFields. java 
// A Class that dynamically adds fields to itself. 
// Demonstrates exception chaining. 

import static net.mindview.util.Print.*; 





class DynamicFieldsException extends Exception {} 
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public class DynamicFields { 
private Objectf][] fields: 
public DynamicFields(int initialSize) { 
fields = new Object[initialSize} [2]: 
for(int 1 = 0; i < initialSize; i++) 
fields[i] = new Object[] { null, null }; 
} 
public String toString() { 
StringBuilder result = new StringBuilder (); 
for (Object{] obj : fields) { 
result. append (obj [9]); 
result.append(": "); 
result. append (obj [1]): 
result.append("\n"); 
} 
return result.tostring(); 
} 
private int hasField(String id) { 
for(int i = @; i < fields. length; i++) 
if (id. equals (fields [i] [9]) 
return i; ， 
return -1; 
} 
private int 
getFieldNumber (String id) throws NoSuchFieldException { 
int fieldNum = hasField(id); 
if (fieldNum == -1) 
throw new NoSuchFieldexception() ; 
return fieldNum; 





} 
private int makeField(String id) { 
for(int i = 0; 1 < fields. length; i++) 
if(fields[i] [0] == null) { 
fields{1] [0] = 14; 
return i; 
} 
// No empty fields. Add one: 
Object[] [] tmp = new Object[fields. length + 1] [2]; 
for(int i = 0; 1 < fields.length; i++) 
tmp[i] = fields[1]; 
for(int i = fields.length; i < tmp.length; 1++) 
tmp[1] = new Object{] { null, null }; 
fields = tmp; 
// Recursive call with expanded fields: x 
return makeField(id) ; 
} 
public Object 
getField(String td) throws NoSuchFieldException { 
return fields [getFieldNumber(id)] [1]; 
} 
public Object setField(String id, Object value) 
throws DynamicFieldsException { 
if(value == null) { 
// Host exceptions don't have a “cause” constructor. 
// In these cases you must use initCause(). 
// available in all Throwable subclasses. 
DynamicFieldsException dfe = 
new DynamicFieldsException(); 
dfe. initCause(new NullPointerException()); 

















throw dfe; 
ie 
int fieldNumber = hasField (id); 
if(fieldNumber == -1) 
fieldNumber = makeField(id); 
Object result = null; 
try { 


result = getField(id); // Get old value : 











466, 














467 
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} catch(NoSuchFieldException e) { 
// Use constructor that takes "cause": 
throw new RuntimeException(e); 


} 
fields[fieldNumber] [1] = value; 
return result; 


} 
public static void main(String[] args) { 

DynamicFields df = new DynamicFields(3); 

print (df); 

try ( 
df.setField("d", "A value for d"); 
df.setField("number™, 47); 
df.setField("number2", 48); 
print(df); 
df.setField("d", "A new value for d"); 
df.setField("number3", 11); 
print("df: " + df); 
print("df.getField(\"d\") : " + df.getField("d")); 
Object field = df.setField("d". null); // Exception 
catch (NoSuchFieldException e) { 
e.printStackTrace(System.out) ; 
catch (DynamicFieldsException e) { 
€.printStackTrace (System. out) ; 
} 


} 
} /* Output: 
null: null 
mull: null 
null: null 








d: A value for d 
number: 47 
number2: 48 


df: d: A new value for d 
number: 47 
number2: 48 
number3: 11 


df.getField("d") : A new value for -d 
DynamicFieldsException 
at DynamicFields. setField(DynanicF ields.java:64) 
at DynamicFields.main(DynamicFields.java:94) 
Caused by: java.lang.NullPointerException 
at DynamicFields. setField(DynamicFields.java:66) 
. 1 more 
S~ 


每 个 DynamicFields 对 象 都 含有 一 个 数组 ， 其 元 素 是 “成 对 的 对 象 "。 第 一 个 对 象 表示 字段 
标识 符 (一 个 字符 串 ) ， 第 二 个 表示 字段 值 ， 值 的 类 型 可 以 是 除 基 本 类 型 外 的 任意 类 型 。 当 创建 


. 对 象 的 时 候 ， 要 合理 估计 一 下 需要 多 少 字段 。 当 调用 setField( 方 法 的 时 候 ， 它 将 试图 通过 标识 


修改 已 有 字段 值 ， 否 则 就 建 一 个 新 的 字段 ， 并 把 值 放 入 。 如 果 空间 不 够 了 ， 将 建立 一 个 更 长 的 
数组 ， 并 把 原来 数组 的 元 素 复制 进去 。 如 果 你 试图 为 字段 设置 一 个 空 值 ， 将 抛 出 一 个 
DynamicFieldsException 异 常 ， 它 是 通过 使 用 initCause0 方 法 把 NullPointerException 对 象 插入 而 
建立 的 。 

至 于 返回 值 ，setField0) 将 用 getField0 方 法 把 此 位 置 的 旧 值 取出 ， 这 个 操作 可 能 会 抛 出 
NoSuchFieldException 异 常 。 如 果 客 户 端 程序 员 调用 了 getField0 方 法 ， 那 么 他 就 有 责任 处 理 这 
个 可 能 抛 出 的 NoSuchFieldException 异 常 ， 但 如 果 异 常 是 从 setField0 方 法 里 抛 出 的 ， 这 种 情况 
将 被 视 为 编程 错误 ， 所 以 就 使 用 接受 cause 参 数 的 构造 器 把 NoSuchFieldException 异 常 转换 为 
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RuntimeException #4 , 

你 会 注意 到 ，toString0 方 法 使 用 了 一 个 StringBuilder 来 创建 其 结果 。 在 第 13 章 中 你 将 会 了 
解 到 更 多 的 关于 StringBuilder 的 知识 ， 但 是 只 要 你 编写 设计 循环 的 toString0 方 法 ， 通 常 都 会 想 
使 用 它 ， 就 像 本 例 一 样 。 

练习 10: (2) 为 一 个 类 定义 两 个 方法 : f0 和 g0。 在 g0 里 ， 抛 出 一 个 自 定义 的 新 异常 。 在 f0 
里 ， 调 用 g0， 捕 获 它 抛 出 的 异常 ， 并 且 在 cateh 子 句 里 抛 出 另 一 个 异常 ( 自 定义 的 第 二 种 异常 )。 
在 main0 里 测试 代码 。 

练习 11: (1) 重复 上 一 个 练习 ， 但 是 在 cateh 子 句 里 把 g0 要 抛 出 的 异常 包装 成 一 个 Runtime 
Exception, 


12.7 Java 标准 异常 


Throwable 这 个 Java 类 被 用 来 表示 任何 可 以 作为 异常 被 抛 出 的 类 。Throwable 对 象 可 分 为 两 
种 类 型 ( 指 从 Throwable 继 承 而 得 到 的 类 型 ) : Error 用 来 表示 编译 时 和 系统 错误 ( 除 特殊 情况 外 ， 
一 般 不 用 你 关心 ) ，Exception 是 可 以 被 抛 出 的 基本 类 型 ， 在 Java 类 库 、 用 户 方法 以 及 运行 时 故 
障 中 都 可 能 抛 出 Exception 型 异常 。 所 以 Java 程 序 员 关心 的 基 类 型 通常 是 Exception 。 

要 想 对 异常 有 全 面 的 了 解 ， 最 好 去 浏览 一 下 HTML 格 式 的 Java 文 档 (可 以 从 java.sun.com 下 
载 )。 为 了 对 不 同 的 异常 有 个 感性 的 认识 ， 这 么 做 是 值得 的 。 但 很 快 你 就 会 发 现 ， 这 些 异常 除了 
名 称 外 其 实 都 差不多 。 同 时 ，Java 中 异常 的 数目 在 持续 增加 ， 所 以 在 书 中 简单 罗列 它们 毫 无 意 
义 。 所 使 用 的 第 三 方 类 库 也 可 能 会 有 自己 的 异常 。 对 异常 来 说 ， 关 键 是 理解 概念 以 及 如 何 使 用 。 

异常 的 基本 的 概念 是 用 名 称 代表 发 生 的 问题 ， 并 且 异 常 的 名 称 应 该 可 以 望 文 知 意 。 异 常 并 
非 全 是 在 javaJang 包 里 定义 的 ， 有 些 异 常 是 用 来 支持 其 他 像 utl、netf 和 io 这 样 的 程序 包 ， 这 些 异 
常 可 以 通过 它们 的 完整 名 称 或 者 从 它们 的 父 类 中 看 出 端倪 。 比 如 ， 所 有 的 输入 /输出 异常 都 是 从 
java.io.IOException 继 承 而 来 的 。 

12.7.1 特例 ，RuntimeException 

在 本 章 的 第 一 个 例子 中 : 


if(t == null) 
throw new NullPointerException(); 


如 果 必 须 对 传递 给 方法 的 每 个 引用 都 检查 其 是 否 为 null (因为 无 法 确定 调用 者 是 否 传人 了 非 
法 引用 )， 这 听 起 来 着 实 吓人 。 幸 运 的 是 ， 这 不 必 由 你 亲自 来 做 ， 它 属于 Java 的 标准 运行 时 检测 
的 一 部 分 。 如 果 对 null 引 用 进行 调用 ，Java 会 自动 抛 出 NullPointerException 异 常 ， 所 以 上 述 代码 
是 多 余 的 ， 尽管 你 也 许 想 要 执行 其 他 的 检查 以 确保 NullPointerException 不 会 出 现 。 

属于 运行 时 异常 的 类 型 有 很 多 ， 它 们 会 自动 被 Java 虚 拟 机 抛 出 ， 所 以 不 必 在 异常 说 明 中 把 
它们 列 出 来 。 这 些 异 常 都 是 从 RuntimeException 类 继承 而 来 ， 所 以 既 体现 了 继承 的 优点 ， 使 用 
起 来 也 很 方便 。 这 构成 了 一 组 具有 相同 特征 和 行为 的 异常 类 型 。 并 且 ， 也 不 再 需要 在 异常 说 明 
中 声明 方法 将 抛 出 RuntimeException 类 型 的 异常 (或 者 任何 从 RuntimeException 继 承 的 异常 ) ， 
它们 也 被 称 为 “不 受 检查 异常 "。 这 种 异常 属于 错误 ， 将 被 自动 捕获 ， 就 不 用 你 亲自 动手 了 。 要 
是 自己 去 检查 RuntimeException 的 话 ， 代 码 就 显得 太 混乱 了 。 不 过 尽管 通常 不 用 捕获 
RuntimeException 异 常 ， 但 还 是 可 以 在 代码 中 抛 出 RuntimeException 类 型 的 异常 。 

如 果 不 捕 获 这 种 类 型 的 异常 会 发 生 什么 事 呢 ? 因为 编译 器 没有 在 这 个 问题 上 对 异常 说 明 进 
行 强制 检查 ，RuntimeException 类 型 的 异常 也 许 会 穿越 所 有 的 执行 路 径直 达 main0 方 法 ， 而 不 
会 被 捕获 。 要 明白 到 底 发 生 了 什么 ， 可 以 试 试 下 面 的 例子 : 





<-> =~ =. T 


264 RIZE 











[470] 














471 








//: exceptions/NeverCaught. java 
// Ignoring Runt imeExceptions. 
/1 {ThrowsException} 


public class NeverCaught { 
static void fO { 
throw new RuntimeException("From f()"); 


} 
static void g() { 
tO; 


Public static void main(String{] args) { 
~ BO: 


} 
} Mi~ 
可 能 读者 已 经 发 现 ，RuntimeException (或 任何 从 它 继承 的 异常 》 是 一 个 特例 。 对 于 这 种 
异常 类 型 ， 编 译 器 不 需要 异常 说 明 ， 其 输出 被 报告 给 了 System.err: 


Exception in thread "main" java.lang.RuntimeException: From f() 
at NeverCaught. f (NeverCaught. java:7) 
at NeverCaugnt.g(NeverCaught. java: 18) 
‘at NeverCaught.main(NeverCaught. java:13) 


所 以 答案 是 ， 如 果 RuntimeException 没 有 被 捕获 而 直达 main0， 那 么 在 程序 退出 前 将 调用 
异常 的 printStackTrace0 方 法 。 

请 务必 记 住 : 只 能 在 代码 中 忽略 RuntimeException (及 其 子 类 ) 类 型 的 异常 ， 其 他 类 型 异 
常 的 处 理 都 是 由 编译 器 强制 实施 的 。 究 其 原因 ，RuntimeException 代 表 的 是 编程 错误 : 

1) 无 法 预料 的 错误 。 比 如 从 你 控制 范围 之 外 传递 进来 的 null 引 用 。 

2) 作为 程序 员 ， 应 该 在 代码 中 进行 检查 的 错误 。( 比 如 对 于 ArrayIndexOutOf- 
BoundsException， 就 得 注意 一 下 数组 的 大 小 了 。) 在 一 个 地 方 发 生 的 异常 ， 常 常会 在 另 一 个 地 
方 导 致 错误 。 

你 会 发 现在 这 些 情况 下 使 用 异常 很 有 好 处 ， 它 们 能 给 调试 带 来 便利 。 

值得 注意 的 是 : 不 应 把 Java 的 异常 处 理 机 制 当 成 是 单一 用 途 的 工具 。 是 的 ， 它 被 设计 用 来 
处 理 一 些 烦人 的 运行 时 错误 ， 这 些 错误 往往 是 由 代码 控制 能 力 之 外 的 因素 导致 的 ， 然 而 ， 它 对 
于 发 现 某 些 编译 器 无 法 检测 到 的 编程 错误 ， 也 是 非常 重要 的 。 

练习 12: (3) 修改 innerclasses/Sequencejava， 使 其 在 你 试图 向 其 中 放置 过 多 地 元 素 时 ， 抛 
出 一 个 合适 的 异常 。 


12.8 使 用 finally 进 行 清理 


对 于 一 些 代码 ， 可 能 会 希望 无 论 try 块 中 的 异常 是 否 抛 出 ， 它 们 都 能 得 到 执行 。 这 通常 适用 
于 内 存 回收 之 外 的 情况 (因为 回收 由 垃圾 回收 器 完成 )。 为 了 达到 这 个 效果 ， 可 以 在 异常 处 理 程 
序 后 面 加 上 finally 子 句 。 完 整 的 异常 处 理 程序 看 起 来 像 这 样 : 


try { 
. // The guarded region: Dangerous activities 
A/ that might throw A, B, or C 

catch(A al) { 

// Handler for situation A 

catch(B b1) { 

// Handler for situation B 

catch(C c1) { 

// Handler for situation C 


O c++ 中 的 异常 处 理 没有 finally 子 句 ， 它 依赖 析 构 函数 来 达到 清理 的 目的 。 
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} finally { 
// Activities that happen every time 
} 


为 了 证 明 finally 子 句 总 能 运行 ， 可 以 试 试 下 面 这 个 程序 : 


/1/: exceptions/FinallyWorks. java 
// The finally clause is always executed. 


class ThreeException extends Exception {} 


public class FinallyWorks { 
static int count = 0; 
public static void main(String{} args) { 
while(true) { 
try { 
// Post-increment is zero first time: 
if(count++ == 6) 
throw new ThreeException(); 
System.out.printin("No exception"); 
} catch(ThreeException e) { 
System. out.printin("ThreeException"); 
} finally { 
System.out.printin("In finally clause"); 
if(count == 2) break; // out of “while” 
$ 
} 


} 
} /* Output: 
ThreeException 
In finally clause 
No exception 
In finally clause 
Wh 


可 以 从 输出 中 发 现 ， 无 论 异 常 是 否 被 抛 出 ，finally 子 句 总 能 被 执行 。 

这 个 程序 也 给 了 我 们 一 些 思路 ， 当 Java 中 的 异常 不 允许 我 们 回 到 异常 抛 出 的 地 点 时 ， 那 么 
该 如 何 应 对 呢 ? 如 果 把 try 块 放 在 循环 里 ， 就 建立 了 一 个 “程序 继续 执行 之 前 必须 要 达到 ”的 条 
件 。 还 可 以 加 入 一 个 static 类 型 的 计数 器 或 者 别 的 装置 ， 使 循环 在 放弃 以 前 能 尝试 一 定 的 次 数 。 
这 将 使 程序 的 健壮 性 更 上 一 个 台阶 。 
12.8.1 finally 用 来 做 什么 

对 于 没有 垃圾 回收 和 析 构 函数 自动 调用 机 制 9 的 语言 来 说 ，finally 非 常 重要 。 它 能 使 程序 员 
TRIE: 无 论 try 块 里 发 生 了 什么 ， 内 存 总 能 得 到 释放 。 但 Java 有 垃圾 回收 机 制 ， 所 以 内 存 释放 不 
再 是 问题 。 而 且 ，Java 也 没有 析 构 函数 可 供 调 用 。 那 么 ，Java 在 什么 情况 下 才能 用 到 finally 呢 ? 

当 要 把 除 内 存 之 外 的 资源 恢复 到 它们 的 初始 状态 时 ， 就 要 用 到 finally 子 句 。 这 种 需要 清理 的 
资源 包括 : 已 经 打开 的 文件 或 网 络 连接 ， 在 屏幕 上 画 的 图 形 ， 甚 至 可 以 是 外 部 世界 的 某 个 开关 ， 
如 下 面 例子 所 示 : 


//: exceptions/Switch. java 
import static net.mindview.util.Print.*; 


public class Switch { 
private boolean state = false: 
public boolean read() { return state; } 
public void on() { state = true; print(this); } 
public void off() { state = false; print(this): } 


日 析 构 函数 是 “ 当 对 象 不 再 被 使 用 的 时 候 ” 会 被 调用 的 函数 。 你 总 能 确切 地 知道 析 构 函数 被 调用 的 时 间 和 地 点 。 
C++ 能 自动 调用 析 构 函数 ， 而 C# ( 它 更 像 Java) 里 面 会 有 自动 进行 清理 的 机 制 。 
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public String toString() { return state ? "on" : "off"; } 
) M11:~ 


// exceptions/OnOffException1.java 
public class OnOffExceptionl extends Exception {} ///:~ 


//: exceptions/On0ffException2. java 
public class OnOffException2 extends Exception {} ///:> 


11: exceptions/On0ffSwitch. java 
// Why use finally? 


public class OnOffSwitch { 

473, private static Switch sw = new Switch(); 
public static void fO 
throws OnOffExceptionl,OnOffException2 {} 
public static void main(String(] args) { 

try { 

Sw.on(); 

// Code that can throw exceptions... 

t0: 

sw.offO; 

catch(OnOffExceptionl e) { 

System. out. printin("OnOffException1") ; 

sw.off(); 

catch(OnOffException2 e) { 

System.out.printin("OnOffException2"); 

SW.off(); 

} 


} 
} /* Output: 











程序 的 目的 是 要 确保 mainO 结 束 的 时 候 开 关 必 须 是 关闭 的 ， 所 以 在 每 个 try 块 和 异常 处 理 程 
序 的 末尾 都 加 入 了 对 sw.off0 方 法 的 调用 。 但 也 可 能 有 这 种 情况 ， 异 常 被 抛 出 ， 但 没 被 处 理 程序 
捕获 ， 这 时 sw.off0 就 得 不 到 调用 。 但 是 有 了 finally， 只 要 把 try 块 中 的 清理 代码 移 放 在 一 处 即 可 ; 


//: exceptions/WithFinally.java 
// Finally Guarantees cleanup. 


public class WithFinally { 
static Switch sw = new Switch(); 
public static void main(String{] args) { 
try { 
sw.on(); 
// Code that can throw exceptions... 
OnoffSwitch.f(); 
} catch(OnOffExceptionl e) { 
System. out.printIn("OnOffException1"); 
catch(OnOffException2 e) { 
System. out .printin("OnOffException2"); 
finally { 
sw.off(); 
474] t 





si 
} /* Output: 


这 里 sw.off0 被 移 到 一 处 ， 并 且 保 证 在 任何 情况 下 都 能 得 到 执行 。 
甚至 在 异常 没有 被 当前 的 异常 处 理 程序 捕获 的 情况 下 ， 异 常 处 理 机 制 也 会 在 跳 到 更 高 一 层 
的 异常 处 理 程序 之 前 ， 执 行 finally 子 句 : 
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//: exceptions/AlwaysFinally. java 
// Finally is always executed. 
import static net.mindview.util.Print.*; 


class FourException extends Exception {} 


public class AlwaysFinally { 
public static void main(String{] args) { 
print("Entering first try block"): 
try { 
print("Entering second try block"): 
try { 
throw new FourException(); 
} finally { 
print("finally in 2nd try block"); 


} 
} catch(FourException e) { 
System. out.printin( 
"Caught FourException in ist try block"); 
} finally { 
System.out.printIn("finally in Ist try block"); 
} 


} 
} /* Output: 
Entering first try block 
Entering second try block 
finally in 2nd try block 
Caught FourException in 1st try block 
finally in Ist try block 
Whim 


当 涉 及 break 和 continue 语 句 的 时 候 ，finally 子 句 也 会 得 到 执行 。 请 注意 ， 如 果 把 finally 子 句 
和 带 标签 的 break 及 continue 配 合 使 用 ， 在 Java 里 就 没 必要 使 用 goto 语 句 了 。 

8313: (2) 修改 练习 9， 加 一 个 finally 子 句 。 验 证 一 下 ， 即 便 是 抛 出 NullPointerException 
异常 ，finally 子 句 也 会 得 到 执行 。 

练习 14: (2) 试 说 明 ， 在 OnOffSwitch.java 的 try 块 内 部 抛 出 RuntimeException， 程 序 可 能 会 
出 现 错误 。 

练习 15: (2) 试 说 明 ， 在 WithFinally.java 的 try 块 内 部 抛 出 RuntimeException， 程 序 不 会 出 
现 错误 。 
12.8.2 在 return 中 使 用 finally 

因为 finally 子 名 总 是 会 执行 的 ， 所 以 在 一 个 方法 中 ， 可 以 从 多 个 点 返回 ， 并 且 可 以 保证 重要 
的 清理 工作 仍旧 会 执行 : 


//: exceptions/Mult ipleReturns. java 
import static net.mindview.util.Print.*; 


public class MultipleReturns { 
public static void f(int i) { 

print("Initialization that requires cleanup"): 
try { 

Print("Point 1"); 

if(1 == 1) return 

print("Point 2"); 

if (i == 2) return: 

print("Point 3"); 

if(i == 3) return; 

print("End") ; 

return; 

finally { 

print("Performing cleanup") ; 
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} 
public static void main(String[{] args) { 
for(int i = 1; 1 <= 4; i++) 
t0): 





} 
|476| } /* Output: 
Initialization that requires cleanup 
Point 1 
Performing cleanup 
Initialization that requires cleanup 
Point 1 
Point 2 
Performing cleanup 
Initialization that requires cleanup 
Point 1 
Point 2 
Point 3 
Performing cleanup 
Initialization that requires cleanup 
Point 1 
Point 2 
Point 3 
End 
Performing cleanup 
“i= 


从 输出 中 可 以 看 出 ， 在 finally 类 内 部 ， 从 何 处 返回 无 关 紧 要 。 

练习 16: (2) 修改 reusing/CADSystemjava， 以 演示 从 try-finally 的 中 间 返 回 仍旧 会 执行 正确 
的 清理 。 

练习 17: (3) 修改 polymorphism/Frog.java， 使 其 使 用 try-finally 来 保证 正确 的 清理 ， 并 展示 
即使 在 try-finally 的 中 间 返 回 ， 它 也 可 以 起 作用 。 
12.8.3 缺憾 异常 丢失 

RHE, Java FERAL LAB. KRABI, RAMAN, HE 
还 是 有 可 能 被 轻易 地 忽略 。 用 某 些 特殊 的 方式 使 用 finally 子 句 ， 就 会 发 生 这 种 情况 : 


//: exceptions/LostMessage. java 
71 Wow an exception can be lost. 











class VeryImportantException extends Exception { 
public String toString() { 
return "A very important exception!”; 
} 





} 


class HoHumException extends Exception { 
public String toString() { 
return "A trivial exception"; 
} 
} 


public class LostMessage { 
void f() throws VeryImportantException { 
throw new VeryImportantException(): 
T 
void dispose() throws HoHumException { 
throw new HoHumException(); 


} 
public static void main(String[] args) { 
try { 
Losthessage Im = new LostMessage(); 
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} finally { 
‘Lm. dispose(); 


} 
} catch(Exception e) { 
System.out.printin(e); 
) 


} Ie Output: 

A trivial exception 

Y~ 

从 输出 中 可 以 看 到 ，VeryImportantException 不 见 了 ， 它 被 finally 子 句 里 的 HoHum- 
Exception 所 取代 。 这 是 相当 严重 的 缺陷 ， 因 为 异常 可 能 会 以 一 种 比 前 面 例子 所 示 更 微妙 和 难以 
察觉 的 方式 完全 丢失 。 相 比 之 下 ，C++ 把 “前 一 个 异常 还 没 处 理 就 抛 出 下 一 个 异常 ”的 情形 看 
成 是 精 糕 的 编程 错误 。 也 许 在 Java 的 未 来 版 本 中 会 修正 这 个 问题 ( 另 一 方面 ， 要 把 所 有 抛 出 异 
常 的 方法 ， 如 上 例 中 的 dispose0 方 法 ， 全 部 打包 放 到 try-catch 子 句 里 面 )。 

一 种 更 加 简单 的 丢失 异常 的 方式 是 从 finally 子 句 中 返回 : 


//: exceptions/ExceptionSilencer. java 


public class ExceptionSilencer { 
public static void main(String[] args) { 
try { 
throw new RuntimeException() ; 
} finatly ¢ 
// Using ‘return’ inside the finally block 
// will silence any thrown exception. 
return; 


} 
} Mi~ 


如 果 运 行 这 个 程序 ， 就 会 看 到 即使 她 出 了 异常 ， 它 也 不 会 产生 任何 输出 。 

练习 18，(3) 为 LostMessagejava 添 加 第 二 层 异 常 丢失 ， 以 便 用 第 三 个 异常 来 替代 HoHum- 
Exception 异常 。 

练习 19，(2) 通过 确保 finally 子 句 中 的 调用 ， 来 修复 LostMessage.java 中 的 问题 。 


12.9 异常 的 限制 


当 和 覆盖 方法 的 时 候 ， 只 能 抛 出 在 基 类 方法 的 异常 说 明 里 列 出 的 那些 异常 。 这 个 限制 很 有 用 ， 
这 意味 着 ， 当 基 类 使 用 的 代码 应 用 到 其 派生 类 对 象 的 时 候 ， 一 样 能 够 工作 (当然 ， 这 是 面 
向 对 象 的 基本 概念 ) ， 异 常 也 不 例外 。 

下 面 例子 演示 了 这 种 (在 编译 时 ) 施加 在 异常 上 面 的 限制 : 


//: exceptions/StormyInning. java 
// Overridden methods may throw only the exceptions 

// specified in their base-class versions, or exceptions 
// derived from the base-class exceptions. 





class BaseballException extends Exception {} 
class Foul extends BaseballException {} 
class Strike extends BaseballException {} 
abstract class Inning { 
public Inning() throws Baseball€xception (} 
Public void event() throws BaseballException { 
// Doesn't actually have to throw anything 
} 
public abstract void atBat() throws Strike, Foul: 
public void walk() {} // Throws no checked exceptions 
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} 


class StormException extends Exception {} 
class RainedOut extends StormException {} 
class PopFoul extends Foul {} 


interface Storm { 
public void event() throws RainedOut; 
public void rainHard() throws RainedOut; 


} 


public class StormyInning extends Inning implements Storm { 
// OK to add new exceptions for constructors, but you 
// must deal with the base constructor exceptions: 
public StormyInning() 
throws RainedOut, Baseballexception {} 
public StormyInning(String s) 
throws Foul, BaseballException {} 
// Regular methods must conform to base class: 
//! void walk() throws PopFoul {} //Compile error 
// Interface CANNOT add exceptions to existing 
// methods from the base class: 
/1/! public void event() throws RainedOut {} 
// If the method doesn't already exist in the 
// base class, the exception is OK: 
public void rainHard() throws RainedOut {} 
// You can choose to not throw any exceptions, 
// even if the base version does: 
public void event() {) 
// Overridden methods can throw inherited exceptions: 
public void atBat() throws PopFoul {} 
Public static void main(String{] args) ( 
try { . 
StormyInning si = new StormyInning(); 
si.atBat(); 
} catch(PopFoul e) { 
System.out.printin("Pop foul"); 
} catch(RainedOut e) { 
System.out.printin("Rained out"); 
} catch(Baseballexception e) { 
System.out.printin(*Generic baseball exception"); 
} 
// Strike not thrown in derived version. 
try ( 
// What happens if you upcast? 
Inning i = new StormyInning(); 
1.atBat(); 
// You must catch the exceptions from the 
// base-class version of the method: 
} catch(Strike e) { 
System. out.printin("Strike"); 
} catch(Foul e) { 
System.out.printin("Foul"): 
} catch(RainedOut e) { 
System.out.printin("Rained out"); 
} catch(BaseballException e) { 
i System.out.println("Generic baseball exception"); 
} 
} Mi~ 


在 Inning 类 中 ， 可 以 看 到 构造 器 和 event0 方 法 都 声明 将 抛 出 异常 ， 但 实际 上 没有 抛 出 。 这 


种 方式 使 你 能 强制 用 户 去 捕获 可 能 在 覆盖 后 的 event0 版 本 中 增加 的 异常 ， 所 以 它 很 合理 。 这 对 
于 抽象 方法 同样 成 立 ， 比 如 atBat0。 


接口 Storm 值 得 注意 ， 因 为 它 包含 了 一 个 在 Inning 中 定义 的 方法 eventO 和 一 个 不 在 Inning 中 
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定义 的 方法 rainHard0。 这 两 个 方法 都 抛 出 新 的 异常 RainedOut。 如 果 StormyInning 类 在 扩展 
Inning 类 的 同时 又 实现 了 Storm 接 口 ， 那 么 Storm 里 的 event() 方 法 就 不 能 改变 在 Inning 中 的 
event0 方 法 的 异常 接口 。 否 则 的 话 ， 在 使 用 基 类 的 时 候 就 不 能 判断 是 否 捕获 了 正确 的 异常 ， 所 
以 这 也 很 合理 。 当 然 ， 如 果 接 口 里 定义 的 方法 不 是 来 自 于 基 类 ， 比 如 rainHard0， 那 么 此 方法 抛 
出 什么 样 的 异常 都 没有 问题 。 

异常 限制 对 构造 器 不 起 作用 。 你 会 发 现 StormyInning 的 构造 器 可 以 抛 出 任何 异常 ， 而 不 必 
理会 基 类 构造 器 所 抛 出 的 异常 。 然 而 ， 因 为 基 类 构造 器 必须 以 这 样 或 那样 的 方式 被 调用 (这 里 
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默认 构造 器 将 自动 被 调用 ) ， 派 生 类 构造 器 的 异常 说 明 必须 包含 基 类 构造 器 的 异常 说 明 。 

派生 类 构造 器 不 能 捕获 基 类 构造 器 抛 出 的 异常 。 

StormyInning.walk0 不 能 通过 编译 的 原因 是 因为 ， 它 抛 出 了 异常 ， 而 Inning.walk0 并 没有 
声明 此 异常 。 如 果 编译 器 允许 这 么 做 的 话 ， 就 可 以 在 调用 Inning.walk0 的 时 候 不 用 做 异常 处 理 
了 ， 而 且 当 把 它 替换 成 Inning 的 派生 类 的 对 象 时 ， 这 个 方法 就 有 可 能 会 抛 出 异常 ， 于 是 程序 就 
失灵 了 。 通 过 强制 派生 类 遵守 基 类 方法 的 异常 说 明 ， 对 象 的 可 替换 性 得 到 了 保证 。 

覆盖 后 的 event0 方 法 表明 ， 派 生 类 方法 可 以 不 抛 出 任何 异常 ， 即 使 它 是 基 类 所 定义 的 异常 。 
同样 这 是 因为 ， 假 使 基 类 的 方法 会 抛 出 异常 ， 这 样 做 也 不 会 破坏 已 有 的 程序 ， 所 以 也 没有 问题 。 
类 似 的 情况 出 现在 atBat0 身 上 ， 它 抛 出 的 是 PopFoul， 这 个 异常 是 继承 自 “ 会 被 基 类 的 atBat0 抛 
出 ”的 Foul。 这 样 ， 如 果 你 写 的 代码 是 同 Inning 打 交道 ， 并 且 调 用 了 它 的 atBat0 的 话 ， 那 么 表 
定 能 捕获 Foul。 而 PopFoul 是 由 Foul 派 生出 来 的 ， 因 此 异常 处 理 程序 也 能 捕获 PopFoul。 

最 后 一 个 值得 注意 的 地 方 是 main0。 这 里 可 以 看 到 ， 如 果 处 理 的 刚好 是 StormyInning 对 象 
的 话 ， 编 译 器 只 会 强制 要 求 你 捕获 这 个 类 所 抛 出 的 异常 。 但 是 如 果 将 它 向 上 转型 成 基 类 型 ， 那 
么 编译 器 就 会 (正确 地 ) 要 求 你 捕获 基 类 的 异常 。 所 有 这 些 限制 都 是 为 了 能 产生 更 为 强壮 的 异 
常 处 理 代码 9 。 

尽管 在 继承 过 程 中 ， 编 译 器 会 对 异常 说 明 做 强制 要 求 ， 但 异常 说 明 本 身 并 不 属于 方法 类 型 
的 一 部 分 ， 方 法 类 型 是 由 方法 的 名 字 与 参数 的 类 型 组 成 的 。 因 此 ， 不 能 基于 异常 说 明 来 重 载 方 
法 。 此 外 ， 一 个 出 现在 基 类 方法 的 异常 说 明 中 的 异常 ， 不 一 定 会 出 现在 派生 类 方法 的 异常 说 明 
里 。 这 点 同 继承 的 规则 明显 不 同 ， 在 继承 中 ， 基 类 的 方法 必须 出 现在 派生 类 里 ， 换 句 话说 ， 在 
继承 和 覆盖 的 过 程 中 ， 某 个 特定 方法 的 “异常 说 明 的 接口 ”不 是 变 大 了 而 是 变 小 了 一 一 这 恰好 
和 类 接口 在 继承 时 的 情形 相反 。 

练习 20，(3) 修改 StormyInningjava， 加 一 个 UmpireArgument 异 常 ， 和 一 个 能 抛 出 此 异常 
的 方法 。 测 试 一 下 修改 后 的 异常 继承 体系 。 


12.10 构造 器 


有 一 点 很 重要 ， 即 你 要 时 刻 询问 自己 “如 果 异 常 发 生 了 ， 所 有 东西 能 被 正确 的 清理 吗 ? ” 
尽管 大 多 数 情况 下 是 非常 安全 的 ， 但 涉及 构造 器 时 ， 问 题 就 出 现 了 。 构 造 器 会 把 对 象 设 置 成 安 
全 的 初始 状态 ， 但 还 会 有 别 的 动作 ， 比 如 打开 一 个 文件 ， 这 样 的 动作 只 有 在 对 象 使 用 完毕 并 且 
用 户 调用 了 特殊 的 清理 方法 之 后 才能 得 以 清理 。 如 果 在 构造 器 内 抛 出 了 异常 ， 这 些 清理 行为 也 
许 就 不 能 正常 工作 了 。 这 意味 着 在 编写 构造 器 时 要 格外 细心 。 

读者 也 许 会 认为 使 用 finally 就 可 以 解决 问题 。 但 问题 并 非 如 此 简单 ， 因 为 finally 会 每 痰 都 执 


© ISO C++ 中 加 上 了 类 似 的 约束 ， 要 求 派生 类 的 方法 所 抛 出 的 异常 要 与 基 类 方法 相同 ， 或 者 是 基 类 方法 抛 出 的 异 
常 的 派生 类 。 这 是 C++ 真 正 能 够 在 编译 时 对 异常 说 明 进行 检查 的 唯一 情况 。 
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行 清理 代码 。 如 果 构 造 器 在 其 执行 过 程 中 半途 而 废 ， 也 许 该 对 象 的 某 些 部 分 还 没有 被 成 功 创建 ， 
而 这 些 部 分 在 finally 子 句 中 却 是 要 被 清理 的 。 

在 下 面 的 例子 中 ， 建 立 了 一 个 InputFile 类 ， 它 能 打开 一 个 文件 并 且 每 次 读 取 其 中 的 一 行 。 
这 里 使 用 了 Java 标 准 输入 /输出 库 中 的 FileReader 和 BufferedReader 类 (将 在 第 18 章 讨论 ) ， 这 些 
类 的 基本 用 法 很 简单 ， 读 者 应 该 很 容易 明白 : 


//: exceptions/InputFile. java 
// Paying attention to exceptions in constructors. 
import java.io.*; 


public class InputFile { 
private BufferedReader in; 
public InputFile(String fname) throws Exception { 
try { 
‘in = new BufferedReader (new FileReader (fname)); 
// Other code that might throw exceptions 
catch(FileNotFoundException e) { 
System.out.printin("Could not open * + fname); 
// Wasn't open, so don't close it 
throw e; 
catch(Exception e) { 
// All other exceptions must close it 
try { 
in.close(); 
} catch(IOException e2) { 
System.out.printin("in.close() unsuccessful"); 


f 

throw e; // Rethrow 
finally { 

// Don't close it here!!! 
} 


} 
public String getLine() { 
String s: 
try { 
5 = in,readLine(); 
} catch(IOException e) { 
throw new Runtime€xception(*readLine() failed"); 
} 


return s; 


} 
public void dispose() { 
try { 
in.close(); 
System.out.printIn("dispose() successful"); 
} catch(I0Exception e2) { 
throw new RuntimeException("in.close() failed"); 


3 
} H= 


InputFile 的 构造 器 接受 字符 串 作为 参数 ， 该 字符 串 表 示 所 要 打开 的 文件 名 。 在 try 块 中 ,会 
使 用 此 文件 名 建立 了 FileReader 对 象 。FileReader 对 象 本 身 用 处 并 不 大 ， 但 可 以 用 它 来 建立 
BufferedReader 对 象 。 注 意 ， 使 用 InputFile 的 好 处 就 能 是 把 两 步 操作 合 而 为 一 。 

如 果 FileReader 的 构造 器 失败 了 ， 将 抛 出 FileNotFoundException 异 常 。 对 于 这 个 异常 ， 并 不 
需要 关闭 文件 ， 因 为 这 个 文件 还 没有 被 打开 。 而 任何 其 他 捕获 异常 的 catch 子 句 必须 关闭 文件 ， 
因为 在 它们 捕获 到 异常 之 时 ， 文 件 已 经 打开 了 (当然 ， 如 果 还 有 其 他 方法 能 抛 出 
FileNotFoundException， 这 个 方法 就 显得 有 些 投 机 取 巧 了 。 这 时 ， 通 常 必须 把 这 些 方法 分 别 放 
到 各 自 的 try 块 里 )。elose0 方 法 也 可 能 会 抛 出 异常 ， 所 以 尽管 它 已 经 在 另 一 个 Catch 子 句 块 里 了 ， 
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还 是 要 再 用 一 县 try-catch 一 一 对 Java 编 译 器 而 言 ， 这 只 不 过 是 又 多 了 一 对 花 括号 。 在 本 地 做 完 处 
理 之 后 ， 异 常 被 重新 抛 出 ， 对 于 构造 器 而 言 这 么 做 是 很 合适 的 ， 因 为 你 总 不 希望 去 误导 调用 方 ， 
让 他 认为 “这 个 对 象 已 经 创建 完毕 ， 可 以 使 用 了 ”。 

在 本 例 中 ， 由 于 finally 会 在 每 次 完成 构造 器 之 后 都 执行 一 遍 ， 因 此 它 实 在 不 该 是 调用 close0 
关闭 文件 的 地 方 。 我 们 希望 文件 在 mputFile 对 象 的 整个 生命 周期 内 都 处 于 打开 状态 。 

getLine( 方 法 会 返回 表示 文件 下 一 行内 容 的 字符 串 。 它 调用 了 能 抛 出 异常 的 readLine0， 但 
是 这 个 异常 已 经 在 方法 内 得 到 处 理 ， 因 此 getLine0 不 会 抛 出 任何 异常 。 在 设计 异常 时 有 一 个 问 
题 : 应 该 把 异常 全 部 放 在 这 一 层 处 理 ， 还 是 先 处 理 一 部 分 ， 然 后 再 向 上 层 抛 出 相同 的 (或 新 的 ) 
异常 又 或 者 是 不 做 任何 处 理 直 接 向 上 层 抛 出 。 如 果 用 法 恰当 的 话 ， 直 接 向 上 层 抛 出 的 确 能 简 
化 编程 。 在 这 里 ，getLine0 方 法 将 异常 转换 为 RuntimeException， 表 示 一 个 编程 错误 。 

用 户 在 不 再 需要 InputFile 对 象 时 ， 就 必须 调用 dispose0 方 法 ， 这 将 释放 BufferedReader 和 / 
或 FileReader 对 象 所 占用 的 系统 资源 (比如 文件 句柄 )， 在 使 用 完 InputFile 对 象 之 前 是 不 会 调 
用 它 的 。 可 能 你 会 考虑 把 上 述 功能 放 到 finalize0 里 面 ， 但 我 在 第 5 章 讲 过 ， 你 不 知道 finalizeO) 
会 不 会 被 调用 (即使 能 确定 它 将 被 调用 ， 也 不 知道 在 什么 时 候 调 用 ) 。 这 也 是 Java 的 缺陷 ， BR 
了 内 存 的 请 理 之 外 ， 所 有 的 清理 都 不 会 自动 发 生 。 所 以 必须 告诉 客户 端 程序 员 ， 这 是 他 们 的 责 
任 。 

对 于 在 构造 阶段 可 能 会 抛 出 异常 ， 并 且 要 求 清理 的 类 ， 最 安全 的 使 用 方式 是 使 用 供 套 的 try 
FA: 

1l: exceptions/Cleanup. java 

// Guaranteeing proper cleanup of a resource. 


public class Cleanup { 
public static void main(String(] args) { 
try { 
InputFile in = new InputFile("Cleanup. java"); 
try { 
String s; 
int i = 1; 
while((s = in.getLine()) != null) 
1 // Perform line-by-line processing here... 
} catch(Exception e) { 
System.out.printin("Caught Exception in main"); 
€.printStackTrace(System.out) ; 
} finally { 
in.dispose(); 


} 
} catch(Exception e) { 

System.out.printin("InputFile construction failed"): 
} 


} 
} /* Output: 
dispose() successful 
“Ii~ 


请 仔细 观察 这 里 的 逻辑 : 对 InputFile 对 象 的 构造 在 其 自己 的 try 语 句 块 中 有 效 ， 如 果 构 造 失 
败 ， 将 进入 外 部 的 catch 子 句 ， 而 dispose( 方 法 不 会 被 调用 。 但 是 ， 如 果 构 造成 功 ， 我 们 肯定 想 
确保 对 象 能 够 被 清理 ， 因 此 在 构造 之 后 立即 创建 了 一 个 新 的 try 语 句 块 。 执 行 清理 的 finally 与 内 
部 的 try 语 句 块 相 关联 。 在 这 种 方式 中 ，finally 子 句 在 构造 失败 时 是 不 会 执行 的 ， 而 在 构成 成 功 
时 将 总 是 执行 。 


这 种 通用 的 清理 惯用 法 在 构造 器 不 抛 出 任何 异常 时 也 应 该 运用 ， 其 基本 规则 是 : 在 创建 需 . 


要 清理 的 对 象 之 后 ， 立 即 进入 一 个 try-finally 语 句 块 : 
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//: exceptions/Cleanupidiom. java 
// Each disposable object must be followed by a try-finally 


class NeedsCleanup { // Construction can't fail 
private static long counter = 1; 
private final long id = counter++; 
public void dispose() { 
System.out.printin("NeedsCleanup " + id + " disposed"); 
} 
$ 


class ConstructionException extends Exception {} 


class NeedsCleanup2 extends NeedsCleanup { 
// Construction can fail: 
public NeedsCleanup2() throws ConstructionException {} 
} 


public class CleanupIdiom { 
public static void main(String(] args) { 
// Section 1: 
NeedsCleanup ncl = new -NeedsCleanup(); 
try { 
H asi 
} finally { 
nc1.dispose() ; 
} 


// Section 2: 
// If construction cannot fail you can group objects: 
NeedsCleanup nc2 = new NeedsCleanup(); 
NeedsCleanup nc3 = new NeedsCleanup() ; 
try { 
ie 
} finally { 
nc3.dispose(); // Reverse order of construction 
nc2.dispose(); 
} 


// Section 3: 
// If construction can fail you must guard each one: 
try { 
NeedsCleanup2 nc4 = new NeedsCleanup2() ; 
try { 
NeedsCleanup2 nc5 = new NeedsCleanup2(): 
try { 
TEs 
} finally { 
nc5.dispose(); 
} 


} catch(ConstructionException. e) { // nc5 constructor 
System.out .println(e); 
} finally { 
ne4.dispose() ; 
} 
} catch(ConstructionException e) { // nc4 constructor 
System.out.printin(e): 
} 
} 
) /* Output: 
NeedsCleanup 1 disposed 
NeedsCleanup 3 disposed 
NeedsCleanup 2 disposed 
NeedsCleanup 5 disposed 
NeedsCleanup 4 disposed 
“I~ 
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在 main0 中 Section 1 相当 简单 : 遵循 了 在 可 去 除 对 象 之 后 紧 跟 try-finally 的 原则 。 如 果 对 
象 构造 不 能 失败 ， 就 不 需要 任何 catch。 在 Section 2 中 ， 为 了 构造 和 清理 ， 可 以 看 到 具有 不 能 失 
败 的 构造 器 的 对 象 可 以 群 组 在 一 起 。 

Section 3 展示 了 如 何 处 理 那些 具有 可 以 失败 的 构造 器 ， 且 需要 清理 的 对 象 。 为 了 正确 处 理 这 
种 情况 ， 事 情 变 得 很 坏 手 ， 因 为 对 于 每 一 个 构造 ， 都 必须 包含 在 其 自己 的 try-finally 语 句 块 中 ， 
并 且 每 一 个 对 象 构造 必须 都 跟随 一 个 try-finally 语 句 块 以 确保 清理 。 

本 例 中 的 异常 处 理 的 坏 手 程度 ， 对 于 应 该 创建 不 能 失败 的 构造 器 是 一 个 有 力 的 论据 ， 尽 管 
这 么 做 并 非 总 是 可 行 。 

注意 ， 如 果 dispose0 可 以 抛 出 异常 ， 那 么 你 可 能 需要 额外 的 try 语 名 块 。 基 本 上 ， 你 应 该 仔 
细 考 虑 所 有 的 可 能 性 ， 并 确保 正确 处 理 每 一 种 情况 。 

练习 21，(2) 试 证 明 ， 派 生 类 的 构造 器 不 能 捕获 它 的 基 类 构造 器 所 抛 出 的 异常 。 

练习 22，(2) 创建 一 个 名 为 FailingConstructorjava 的 类 ， 它 具有 一 个 可 能 会 在 构造 过 程 中 失 
败 并 且 会 抛 出 一 个 异常 的 构造 器 。 在 main0 中 ， 编 写 能 够 确保 不 出 现 故障 的 代码 。 

练习 23，(4) 在 前 一 个 练习 中 添加 一 个 dispose0 方 法 。 修 改 FailingConstructor， 使 其 构造 器 
可 以 将 那些 可 去 除 对 象 之 一 当 作 一 个 成 员 对 象 创建 ， 然 后 该 构造 器 可 能 会 抛 出 一 个 异常 ， 之 后 
它 将 创建 第 二 个 可 去 除 成 员 对 象 。 编 写 能 够 确保 不 出 现 故 障 的 代码 ， 并 在 main0 中 验证 所 有 可 
能 的 故障 情形 都 被 种 盖 了 。 

练习 24: (2) 在 FailingConstructor 类 中 添加 一 个 dispose0 方 法 ,并 编写 代码 正确 使 用 这 个 类 。 


12.11 异常 匹配 


抛 出 异常 的 时 候 ， 异 常 处 理 系统 会 按照 代码 的 书写 顺序 找 出 “最 近 ” 的 处 理 程序 。 找 到 匹 
配 的 处 理 程序 之 后 ， 它 就 认为 异常 将 得 到 处 理 ， 然 后 就 不 再 继续 查找 。 

查找 的 时 候 并 不 要 求 抛 出 的 异常 同 处 理 程序 所 声明 的 异常 完全 匹配 。 派 生 类 的 对 象 也 可 以 
匹配 其 基 类 的 处 理 程序 ， 就 像 这 样 : 


//: exceptions/Human. java 
// Catching exception hierarchies. 


class Annoyance extends Exception {} 
class Sneeze extends Annoyance {} 


public class Human { 
public static void main(String{) args) { 

// Catch the exact type: 

try { 
throw new Sneeze(): 

} catch(Sneeze s) { 
System.out.printin("Caught Sneeze"); 

} catch(Annoyance a) { 
System.out.printin("Caught Annoyance") ; 


了 

// Catch the base type: 

try { 
throw new Sneeze(); 

} catch(Annoyance a) { 
System.out.printin("Caught Annoyance"); 


} 
}/* Output: 
Caught Sneeze 
Caught Annoyance 
Whim 
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Sneeze 异 常会 被 第 一 个 匹配 的 cateh 子 句 捕获 ， 也 就 是 程序 里 的 第 一 个 。 然 而 如 果 将 这 个 
catch 子 句 删 掉 ， 只 留 下 Annoyance 的 catch 子 句 ， 该 程序 仍然 能 运行 ， 因 为 这 次 捕获 的 是 Sneeze 
的 基 类 。 换 名 话说 ，catch(Annoyance e) 会 捕获 Annoyance 以 及 所 有 从 它 派生 的 异常 。 这 一 点 非 
常 有 用 ， 因 为 如 果 决 定 在 方法 里 加 上 更 多 派生 异常 的 话 ， 只 要 客户 程序 员 捕 获 的 是 基 类 异常 ， 
那么 它们 的 代码 就 无 需 更 改 。 

如 果 把 捕获 基 类 的 catch 子 句 放 在 最 前 面 ， 以 此 想 把 派生 类 的 异常 全 给 “屏蔽 ” 掉 ， 就 像 
这 样 ; 

try { 

throw new Sneeze(); 
} catch(Annoyance a) { 
H ses 
} catch(Sneeze s) { 


Wey 
} 


这 样 编译 器 就 会 发 现 Sneeze 的 catch 子 句 永远 也 得 不 到 执行 ， 因 此 它 会 向 你 报告 错误 。 

练习 25: (2) 建立 一 个 三 层 的 异常 继承 体系 ， 然 后 创建 基 类 A， 它 的 一 个 方法 能 抛 出 异常 体 
系 的 基 类 异常 。 让 B 继 承 A， 并 且 和 覆盖 这 个 方法 ， 让 它 抛 出 第 二 层 的 异常 。 让 C 继 承 B， 再 次 覆盖 
这 个 方法 ， 让 它 抛 出 第 三 晨 的 异常 。 在 main0 里 创建 一 个 C 类 型 的 对 象 ， 把 它 向 上 转型 为 A， 然 
后 调用 这 个 方法 。 


12.12 其 他 可 选 方式 


异常 处 理 系统 就 像 一 个 活 门 〈trap door) ， 使 你 能 放弃 程序 的 正常 执行 序列 。 当 “异常 情形 ” 
发 生 的 时 候 ， 正 常 的 执行 已 变 得 不 可 能 或 者 不 需要 了 ， 这 时 就 要 用 到 这 个 “ 活 门 "。 蜡 常 代表 了 
当前 方法 不 能 继续 执行 的 情形 。 开 发 异常 处 理 系统 的 原因 是 ， 如 果 为 每 个 方法 所 有 可 能 发 生 的 
错误 都 进行 处 理 的 话 ， 任 务 就 显得 过 于 繁重 了 ， 程 序 员 也 不 愿意 这 么 做 。 结 果 常 常 是 将 错误 忽 
赂 。 应 该 注意 到 ， 开 发 异常 处 理 的 初衷 是 为 了 方便 程序 员 处 理 错误 。 

异常 处 理 的 一 个 重要 原则 是 “只 有 在 你 知道 如 何 处 理 的 情况 下 才 捕 获 异 常 "。 实 际 上 ， 异 常 
处 理 的 一 个 重要 目标 就 是 把 错误 处 理 的 代码 同 错误 发 生 的 地 点 相 分 离 。 这 使 你 能 在 一 段 代 码 中 
专注 于 要 完成 的 事情 ， 至 于 如 何 处 理 错误 ， 则 放 在 另 一 段 代码 中 完成 。 这 样 以 来 ， 主 于 代码 就 
不 会 与 错误 处 理 逻 辑 混在 一 起 ， 也 更 容易 理解 和 维护 。 通 过 允许 一 个 处 理 程序 去 处 理 多 个 出 错 
点 ， 异 常 处 理 还 使 得 错误 处 理 代码 的 数量 趋向 于 减少 。 

“被 检查 的 异常 ”使 这 个 问题 变 得 有 些 复杂 ， 因 为 它们 强制 你 在 可 能 还 没准 备 好 处 理 错误 的 
时 候 被 迫 加 上 cateh 子 名 ， 这 就 导致 了 知 食 则 有 害 (harmful if swallowed) 的 问题 : 

try { 


II ... to do something useful 
} catch(ObligatoryException e) {} // Gulp! 


程序 员 们 只 做 最 简单 的 事情 (包括 我 自己 ， 在 本 书 第 1 版 中 也 有 这 个 问题 )， 常 常 是 无 意 中 
“吞食 ”了 异常 ， 然 而 一 旦 这 么 做 ， 虽 然 能 通过 编译 ， 但 除非 你 记得 复查 并 改正 代码 ， 否 则 异常 
将 会 丢失 。 异 常 确实 发 生 了 ， 但 “吞食 ”后 它 却 完全 消失 了 。 因 为 编译 器 强迫 你 立刻 写 代 码 来 
处 理 异 常 ， 所 以 这 种 看 起 来 最 简单 的 方法 ， 却 可 能 是 最 糟糕 的 做 法 。 

当 我 意识 到 犯 了 这 么 大 一 个 错误 时 ， 简 直 吓 了 一 大 跳 。 在 本 书 第 2 版 中 ， 我 在 处 理 程序 里 通 
过 打印 栈 轨迹 的 方法 “修补 ”了 这 个 问题 (本章 中 的 很 多 例子 还 是 使 用 了 这 种 方法 ， 看 起 来 还 
是 比较 合适 的 )。 虽 然 这 样 可 以 跟踪 异常 的 行为 ， 但 是 仍旧 不 知道 该 如 何 处 理 异常 。 这 一 节 ， 我 
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们 来 研究 一 下 “被 检查 的 异常 ”及 其 并 发 症 ， 以 及 采用 什么 方法 来 解决 这 些 问题 。 

这 个 话题 看 起 来 简单 ， 但 实际 上 它 不 仅 复杂 ， 更 重要 的 是 还 非常 多 变 。 总 有 人 会 顽固 地 坚 
持 自己 的 立场 ， 声 称 正确 答案 (也 是 他 们 的 答案 ) 是 显而易见 的 。 我 觉得 之 所 以 会 有 这 种 观点 ， 
是 因为 我 们 使 用 的 工具 已 经 不 是 ANSI 标 准 出 台 前 的 像 C 那 样 的 弱 类 型 语言 ， 而 是 像 Ct++ 和 Java 这 
样 的 “ 强 静 态 类 型 语言 ”( 也 就 是 编译 时 就 做 类 型 检查 的 语言 )， 这 是 前 者 所 无 法 比拟 的 。 当 刚 
开始 这 种 转变 的 时 候 (就 像 我 一 样 )， 会 觉得 它 带 来 的 好 处 是 那样 明显 ， 好 像 类 型 检查 总 能 解决 
所 有 的 问题 。 在 此 ， 我 想 结合 我 自己 的 认识 过 程 ， 告 诉 读者 我 是 怎样 从 对 类 型 检查 的 绝对 迷信 
变 成 持 怀疑 态度 的 ， 当 然 ， 很 多 时 候 它 还 是 非常 有 用 的 ， 但 是 当 它 挡住 我 们 的 去 路 并 成 为 障碍 
的 时 候 ， 我 们 就 得 跨 过 去 。 只 是 这 条 界限 往往 并 不 是 很 清晰 (我 最 喜欢 的 一 句 格言 是 ， 所 有 模 
型 都 是 错误 的 ， 但 有 些 是 能 用 的 ) 。 

12.12.1 历史 

异常 处 理 起 源 于 PL/1 和 Mesa 之 类 的 系统 中 ， 后 来 又 出 现在 CLU、Smalltalk、Modula-3、Ada、 
Eiffel、C++、Python、Java 以 及 后 Java 语 言 Ruby 和 C# 中 。Java 的 设计 和 C++ 很 相似 ， 只 是 Java 的 
设计 者 去 掉 了 一 些 他 们 认为 C++ 设计 得 不 好 的 东西 。 

为 了 能 向 程序 员 提供 一 个 他 们 更 愿意 使 用 的 错误 处 理 和 恢复 的 框架 ， 异 常 处 理 机 制 很 晚 才 
被 加 入 C++ 标准 化 过 程 中 ， 这 是 由 C++ 的 设计 者 Bjarme Stroustrup 所 倡议 的 。C++ 的 异常 模型 主要 
借鉴 了 CLU 的 做 法 。 然 而 ， 当 时 其 他 语言 已 经 支持 异常 处 理 了 : 包括 Ada、Smalltalk (两 者 都 有 
异常 处 理 ， 但 是 都 没有 异常 说 明 )， 以 及 Modula-3 ( 它 既 有 异常 处 理 也 有 异常 说 明 )。 

Liskov 和 Snyder 在 他 们 的 一 篇 讨论 该 主题 的 独创 性 论文 中 指出 ， 用 瞬时 风格 (transient 
fashion) 报告 错误 的 语言 (如 C 中 ) 有 一 个 主要 缺陷 ， 那 就 是 ; 

“………" 每 次 调用 的 时 候 都 必须 执行 条 件 测试 ， 以 确定 会 产生 何 种 结果 。 这 使 程序 难以 阅读 ， 
并 且 有 可 能 降低 运行 效率 ， 因 此 查 序 员 们 既 不 愿意 指出 ， 也 不 愿意 处 理 异 常 。” 

因此 ， 异 常 处 理 的 初 囊 是 要 消除 这 种 限制 ， 但 是 我 们 又 从 Java 的 “被 检查 的 异常 ”中 看 到 
了 这 种 代码 。 他 们 继续 写 道 : 

要 求 程序 员 把 异常 处 理 程序 的 代码 文本 附 接 到 会 引发 异常 的 调用 上 ， 这 会 降低 程序 的 
可 读 性 ， 使 得 程序 的 正常 思路 被 异常 处 理 给 破坏 了 。” 

C++ 中 异常 的 设计 参考 了 CLU 方 式 。Stroustrup 声 称 其 目标 是 减少 恢复 错误 所 需 的 代码 。 我 
想 他 这 话 是 说 给 那些 通常 情况 下 都 不 写 C 的 错误 处 理 的 程序 员 们 听 的 ， 因 为 要 把 那么 多 代码 放 到 
那么 多 地 方 实在 不 是 什么 好 差事 。 所 以 他 们 写 C 程 序 的 习惯 是 ， 忽 略 所 有 的 错误 ， 然 后 使 用 调试 
器 来 跟踪 错误 。 这 些 程序 员 知 道 ， 使 用 异常 就 意味 着 他 们 要 写 一 些 通常 不 用 写 的 、“ 多 出 来 的 ” 
代码 。 B 
评价 Java 的 “被 检查 的 异常 ”的 时 候 ， 这 一 点 是 很 重要 的 。 

C++ 从 CLU 那 里 还 带 来 另 一 种 思想 : 异常 说 明 。 这 样 ， 就 可 以 用 编程 的 方式 在 方法 的 特征 
签名 中 ， 声 明 这 个 方法 将 会 抛 出 异常 。 异常 说 明 可 能 有 两 种 意思 。 一 个 是 “我 的 代码 会 产生 这 
种 异常 ， 这 由 你 来 处 理 ”"。 另 一 个 是 “我 的 代码 忽略 了 这 些 异 常 ， 这 由 你 来 处 理 ”。 学 习 异 常 处 
理 的 机 制 和 语法 的 时 候 ， 我 们 一 直 在 关注 “你 来 处 理 ”部 分 ， 但 这 里 特别 值得 注意 的 事实 是 ， 
我 们 通常 都 忽略 了 异常 说 明 所 表达 的 完整 含义 。 

C++ 的 异常 说 明 不 属于 函数 的 类 型 信息 。 编 译 时 唯一 要 检查 的 是 异常 说 明 是 不 是 前 后 一 致 ， 
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， 要 把 他 们 拉 到 “使 用 错误 处 理 ” 的 正轨 上 ,“ 多 出 来 的 ”代码 决 不 能 太 多 。 我 认为 ， A 
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比如 ， 如 果 函 数 或 方法 会 抛 出 某 些 异常 ， BBL EAT MRL wR Re Ti A A as DE 
常 。 与 Java 不 同 ，C++ 不 会 在 编译 时 进行 检查 以 确定 函数 或 方法 是 不 是 真 的 抛 出 异常 或 者 异常 
说 明 是 不 是 完整 (也 就 是 说 ， 异 常 说 明 有 没有 精确 描述 所 有 可 能 被 抛 出 的 异常 )。 这 样 的 检查 只 
发 生 在 运行 期 间 。 如 果 抛 出 的 异常 与 异常 说 明 不 符 ，C++ 会 调用 标准 类 库 的 unexpected0 函 数 。 

值得 注意 的 是 ， 由 于 使 用 了 模板 ，C++ 的 标准 类 库 实现 里 根本 没有 使 用 异常 说 明 。 在 Java 中 ， 
对 于 范 型 用 于 异常 说 明 的 方式 存在 着 一 些 限制 。 
12.12.2 观点 

首先 ，Java 无 谓 地 发 明了 “被 检查 的 异常 ”( 很 明显 是 受 C++ 异 常 说明 的 启发 ， 以 及 受 C++ 
程序 员 们 一 般 对 此 无 动 于 圳 的 事实 的 影响 ) 。 但 是 ， 这 还 只 是 一 次 尝试 ， 目 前 为 止 还 没有 别 的 语 
言 采 用 这 种 做 法 。 

其 次 ， 仅 从 示意 性 的 例子 和 小 程序 来 看 ,“ 被 检查 的 异常 ”的 好 处 很 明显 。 但 是 当 程序 开始 
变 大 的 时 候 ， 就 会 带 来 一 些微 妙 的 问题 。 当 然 ， 程 序 不 是 一 下 就 变 大 的 ， 这 有 个 过 程 。 如 果 把 
不 适用 于 大 项 目的 语言 用 于 小 项 目 ， 当 这 些 项 目 不 断 膨胀 时 ， 突 然 有 一 天 你 会 发 现 ， 原 来 可 以 
管理 的 东西 ， 现 在 已 经 变 得 无 法 管理 了 。 这 就 是 我 所 说 的 过 多 的 类 型 检查 ， 特 别 是 “被 检查 的 
异常 ”所 造成 的 问题 。 

看 来 程序 的 规模 是 个 重要 因素 。 由 于 很 多 讨论 都 用 小 程序 来 做 演示 ， 因 此 这 并 不 足以 说 明 
问题 。 一 名 C# 的 设计 人 员 发 现 : 

“ 仅 从 小 程序 来 看 ， 会 认为 异常 说 明 能 增加 开发 人 员 的 效率 ， 并 提高 代码 的 质量 ;但 考察 大 
项 目的 时 候 ， 结 论 或 不 同 了 一 一 开发 效率 下 降 了， 而 代码 质量 只 有 微不足道 的 提高 ， 甚 至 毫 无 
ih", © 

谈 到 未 被 捕获 的 异常 的 时 候 ，CLU 的 设计 师 们 认为 : 

“我 们 觉得 强迫 程 序 员 在 不 知道 该 采取 什么 措施 的 时 候 提供 处 理 程序 ， 是 不 现实 的 ”日 

在 解释 为 什么 “函数 没有 异常 说 明 就 表示 可 以 抛 出 任何 异常 ”的 时 候 ，Stroustrup 这 样 认为 

“但 是 ， 这 样 一 来 几乎 所 有 的 函数 都 得 提供 异常 说 明了 ， 也 就 都 得 重新 编译 ， 而 且 还 会 妨碍 
它 同 其 他 语言 的 交互 。 这 样 会 迫使 程序 员 速 反 异 常 处 理 机 制 的 约束 ， 他 们 会 写 欺 骗 程 序 来 拖 盖 
异常 。 这 将 给 没有 注意 到 这 些 异 常 的 人 造成 一 种 诬 假 的 安全 感 。” 昌 

我 们 已 经 看 到 这 种 破坏 异常 机 制 的 行为 了 一 一 就 在 Java 的 “被 检查 的 异常 ”里 。 

Martin Fowler (UML Distilled，Refactoring 和 Analysis Patterns 的 作者 ) 给 我 写 了 下 面 这 段 话 : 

“…… 总 体 来 说 ， 我 觉得 异常 很 不 错 ， 但 是 Java 的 ”被 检查 的 异常 “ 带 来 的 麻烦 比 好 处 要 多 。 

我 觉得 Java 的 当务之急 应 该 是 统一 其 报告 错误 的 模型 ， 这 样 所 有 的 错误 都 能 通过 异常 来 报 
告 。C++ 不 这 么 做 的 原因 是 它 要 考虑 向 后 兼容 ， 要 照顾 那些 直接 忽略 所 有 错误 的 C 代 码 。 但 是 如 
果 你 一 致 地 用 异常 来 报告 错误 ， 那 么 只 要 愿意 ， 随 时 可 以 抛 出 异常 ， 如 果 不 愿意 ， 这 些 错误 会 
被 传播 到 最 上 层 (控制 台 或 其 他 容器 程序 )。 只 有 当 Java 修 改 了 它 那 类 似 C++ 的 模型 ， 使 异常 成 
为 报告 错误 的 唯一 方式 ， 那 时 “被 检查 的 异常 ”的 额外 限制 也 许 就 会 变 得 没有 那么 必要 了 。 

过 去 ， 我 曾 坚定 地 认为 “被 检查 的 异常 ”和 强 静态 类 型 检查 对 开发 健壮 的 程序 是 非常 必要 
的 。 但 是 ,我 看 到 的 以 及 我 使 用 一 些 动态 (类 型 检查 ) 语言 的 亲身 经 历 ? 告 诉 我 ， 这 些 好 处 实际 


© httpi//discuss.develop.com/archives/wa.exe?A2=indO011 A&L=DOTNET&P=R32820,, 


© http://discuss.develop.com/archives/wa.exe?A2=ind0011A&L=DOTNET&P=R32820, 

© Bjame Stroustrup, The C++ Programming Language, 3rd edition 【C++ 程序 设计 语言 ， 第 3 版 )，Addison-Wesley 1997, 
第 376 页 。 

@ 间接 经 验 来 自 于 与 很 多 资深 Smalitalk 程 序 员 的 谈话 ， 直 接 经 验 则 得 自 于 使 用 Python (www.Python.org)。 
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上 是 来 自 于 : 

D 不 在 于 编译 器 是 否 会 强制 程序 员 去 处 理 错误 ， 而 是 要 有 一 致 的 、 使 用 异常 来 报告 错误 的 
模型 。 
2) 不 在 于 什么 时 候 进行 检查 ， 而 是 一 定 要 有 类 型 检查 。 也 就 是 说 ， 必 须 强制 程序 使 用 正确 
的 类 型 ， 至 于 这 种 强制 施加 于 编译 时 还 是 运行 时 ， 那 倒 没 关系 。 

此 外 ,减少 编译 时 施加 的 约束 能 显著 提高 程序 员 的 编程 效率 。 事 实 上， 反射 和 泛 型 就 是 用 来 
补偿 静态 类 型 检查 所 带 来 的 过 多 限制 ， 在 本 书 很 多 例子 中 都 会 见 到 这 种 情形 。 

我 已 经 听 到 有 人 在 指责 了 ， 他 们 认为 这 种 言论 会 令 我 名 誉 扫地 ， 会 让 文明 堕落 ， 会 导致 更 
高 比例 的 项 目 失 败 。 他 们 的 信念 是 应 该 在 编译 时 指出 所 有 错误 ， 这 样 才能 挽救 项 目 ， 这 种 信念 
可 以 说 是 无 比 坚定 的 ， 其 实 更 重要 的 是 要 理解 编译 器 的 能 力 限制 。 在 http://MindView.net/ 
Books/BetterJava 上 的 补充 材料 中 ， 我 强调 了 自动 构建 过 程 和 单元 测试 的 重要 性 ， 比 起 把 所 有 的 
东西 都 说 成 是 语法 错误 ， 它 们 的 效果 可 以 说 是 事半功倍 。 下 面 这 段 话 是 至 理 名 言 ; 

好 的 程序 设计 语言 能 帮助 程序 员 写 出 好 程序 ， 但 无 论 哪 种 语言 都 避免 不 了 程序 员 用 它 写 出 
TRUER, ° 
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语言 来 说 ， 这 个 变化 可 能 太 激 进 了 点 ， 况 且 Sun 的 支持 者 们 也 非常 强大 。Sun 有 完全 向 后 兼容 的 
历史 和 策略 ， 实 际 上 所 有 Sun 的 软件 都 能 在 Sun 的 硬件 上 运行 ， 无 论 它们 有 多 么 古老 。 然 而 ， 如 
果 发 现 有 些 “ 被 检查 的 异常 ”挡住 了 路 ， 尤 其 是 发 现 你 不 得 不 去 对 付 那些 不 知道 该 如 何 处 理 的 
异常 ， 还 是 有 些 办 法 的 。 

12.123 把 异常 传递 给 控制 台 

对 于 简单 的 程序 ， 比 如 本 书 中 的 许多 例子 ， 最 简单 而 又 不 用 写 多 少 代码 就 能 保护 异常 信息 
的 方法 ， 就 是 把 它们 从 main0 传 递 到 控制 台 。 例 如 ， 为 了 读 取信 息 而 打开 一 个 文件 (在 第 12 章 
将 详细 介绍 )， 必 须 对 FileInputStream 进 行 打开 和 关闭 操作 ， 这 就 可 能 会 产生 异常 。 对 于 简单 的 
程序 ， 可 以 像 这 样 做 《本 书 中 很 多 地 方 采用 了 这 种 方法 ) « 

/1/: exceptions/MainException. java 

import java.io.*; 


public class MainException { 
// Pass all exceptions to the console: 
public static void main(String{) args) throws Exception { 
// Open the file: 
FileInputStream file = 
new FileInputStream("MainException. java"); 
// Use the file ... 
// Close the file: 
file.close(); 


} 
} Ii~ 
注意 ，main0 作 为 一 个 方法 也 可 以 有 异常 说 明 ， 这 里 异常 的 类 型 是 Exception， 它 也 是 所 有 
“被 检查 的 异常 ”的 基 类 。 通 过 把 它 传递 到 控制 台 , 就 不 必 在 main0 里 写 try-catch 子 句 了 。( 不 过 ， 
实际 的 文件 输入 /输出 操作 比 这 个 例子 要 复杂 得 多 ， 所 以 在 学 习 第 18 章 之 前 ， 别 高 兴 得 太 早 。) 
12.12.4 把 “被 检查 的 异常 ”转换 为 “不 检查 的 异常 ” 
在 编写 你 自己 使 用 的 简单 程序 时 ， 从 main0 中 抛 出 异常 是 很 方便 的 ， 但 这 不 是 通用 的 方法 。 


© (Kees Koster, CDL 语 言 的 设计 者 ， 引 自 Eiffel 语 言 的 设计 者 Bertrand Meyer。) http://www.elj.com/elj/vi/n1/ 
bmvright/, 
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问题 的 实质 是 ， 当 在 一 个 普通 方法 里 调用 别 的 方法 时 ， 要 考虑 到 “我 不 知道 该 这 样 处 理 这 个 异 
常 ， 但 是 也 不 想 把 它 “ 知 ”了 ， 或 者 打印 一 些 无 用 的 消息 "。JDK 1.4 的 异常 链 提供 了 一 种 新 的 
思路 来 解决 这 个 问题 。 可 以 直接 把 “被 检查 的 异常 ”包装 进 RuntimeException 里 面 ， 就 像 这 样 ; 





1/ ... to do something useful 

} catch(IDontknowWhatToDoWi thThisCheckedException e) { 
throw new RuntimeException(e); 

} 


如 果 想 把 “被 检查 的 异常 ”这 种 功能 “ 屏 项” 掉 的 话 ， 这 看 上 去 像 是 一 个 好 办 法 。 不 用 
“ 吞 下 ”异常 ， 也 不 必 把 它 放 到 方法 的 异常 说 明 里 面 ， 而 异常 链 还 能 保证 你 不 会 丢失 任何 原始 异 
常 的 信息 。 

这 种 技巧 给 了 你 一 种 选择 ， 你 可 以 不 写 try-cateh 子 句 和 /或 异常 说 明 ， 直 接 忽略 异常 ， 让 它 
自己 沿 着 调用 栈 往 上 “ 冒 泡 "。 同 时 ， 还 可 以 用 getCause0 捕 获 并 处 理 特定 的 异常 ， 就 像 这 样 


//: exceptions/TurnOf fChecking. java 

// "Turning off" Checked exceptions. 
import java. io.*; 

import static net.mindview.util.Print.*; 


class WrapCheckedException { 
void throwRuntimeException(int type) { 
try { 
, switch(type) { 

case 0: throw new FileNotFoundException(); 
case 1: throw new IOException(); 
case 2: throw new RuntimeException("Where am 12"); 
default: return; 


} 
} catch(Exception e) { // Adapt to unchecked: 
throw new RuntimeException(e) ; 


} 
} 


class SomeOtherException extends Exception {} 





public class TurnOffChecking { 

public static void main(String{) args) { 
WrapCheckedException wce = new WrapCheckedException() ; 
// You can call throwRuntimeException() without a try 

` // block, and let RuntimeExceptions leave the method: 
wee. throwRunt imeException(3); 
// Or you can choose to catch exceptions: 
for(int 1 = 0; 1 < 4; i++) 


try { 
if < 3) 
wee. throwRunt imeException(i); 
else 


throw new SomeOtherException(); 
catch (SomeOtherException e) { 

print("SomeOtherException: " + e); 
catch(RuntimeException re) { 
try { 

throw re.getCause(); 
} catch(FileNotFoundexception e) { 
print ("FileNotFoundException: ”+ e); 
catch(IOException e) { 
print("I0Exception: ”+ e); 
catch(Throwable e) { 
print("Throwable: " + e); 
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} 


} 
} /* Output: 
FileNotFoundException: java.io.FileNotFoundException 
IOException: java.io. IOException 
Throwable: java. lang.RuntimeException: Where am I? 
SomeOtherException: SomeOtherException 
Wham 


WrapCheckedException.throwRuntimeException0 的 代码 可 以 生成 不 同类 型 的 异常 。 这 些 
异常 被 捕获 并 包装 进 了 RuntimeException 对 象 ， 所 以 它们 成 了 这 些 运行 时 异常 的 “cause” 了 。 

在 TurnOffChecking 里 ， 可 以 不 用 try 块 就 调用 throwRuntimeException0， 因 为 它 没有 抛 
出 “被 检查 的 异常 "。 但 是 ， 当 你 准备 好 去 捕获 异常 的 时 候 ， 还 是 可 以 用 try 块 来 捕获 任何 你 想 
捕获 的 异常 的 。 应 该 捕获 try 块 肯定 会 抛 出 的 异常 ， 这 里 就 是 SomeOtherException。Runtime- 
了 Exception 要 放 到 最 后 去 捕获 。 然 后 把 getCause0 的 结果 (也 就 是 被 包装 的 那个 原始 异常 ) 抛 出 
来 。 这 样 就 把 原先 的 那个 异常 给 提取 出 来 了 ， 然 后 就 可 以 用 它们 自己 的 catch 子 句 进行 处 理 。 

本 书 余下 部 分 将 会 在 合适 的 时 候 使 用 这 种 “用 RuntimeException 来 包装 “被 检查 的 异常 '” 
的 技术 。 另 一 种 解决 方案 是 创建 自己 的 RuntimeException 的 子 类 。 在 这 种 方式 中 ， 不 必 捕 获 它 ， 
但 是 希望 得 到 它 的 其 他 代码 都 可 以 捕获 它 。 

练习 27:(1) 修改 练习 3， 将 异常 转变 为 RuntimeException。 

练习 28: (1) 修改 练习 4， 使 客户 的 异常 类 继承 自 RuntimeException， 并 展示 编译 器 允许 你 
省 略 try 语 句 块 。 

练习 29，(D 修改 StormyInningjava 中 所 有 的 异常 类 型 ， 使 它们 护 展 RuntimeException， 并 
展示 这 里 不 需要 任何 异常 说 明 或 try 语 句 块 。 移 除 “/!” 注 释 并 展示 这 些 方法 不 需要 说 明 就 可 以 
编译 。 

练习 30，(2) 修改 Human.java， 使 异常 继承 自 RuntimeException 修 改 main()， 使 其 用 
TurnOfrCheckingjava 类 处 理 不 同类 型 的 异常 。 


12.13 异常 使 用 指南 


应 该 在 下 列 情况 下 使 用 异常 : 

1) 在 恰当 的 级 别处 理 问题 。( 在 知道 该 如 何 处 理 的 情况 下 才 捕 获 异常 。) 

2) 解决 问题 并 且 重 新 调用 产生 异常 的 方法 。 

3) 进行 少许 修补 ， 然 后 绕 过 异常 发 生 的 地 方 继续 执行 。 

4) 用 别 的 数据 进行 计算 ， 以 代替 方 法 预计 会 返回 的 值 。 

5) 把 当前 运行 环境 下 能 做 的 事情 尽量 做 完 ， 然 后 把 相同 的 异常 重 抛 到 更 高 层 。 

6) 把 当前 运行 环境 下 能 做 的 事情 尽量 做 完 ， 然 后 把 不 同 的 异常 抛 到 更 高 层 。 

7) 终止 程序 。 

8) 进行 简化 。( 如 果 你 的 异常 模式 使 问题 变 得 太 复杂 ， 那 用 起 来 会 非常 痛苦 也 很 烦人 。) 

多 让 类 库 和 程序 更 安全 。( 这 既是 在 为 调试 做 短期 投资 ， 也 是 在 为 程序 的 健壮 性 做 长 期 投资 。) 


12.14 总 结 


异常 是 Java 程 序 设计 不 可 分 割 的 一 部 分 ， 如 果 不 了 解 如 何 使 用 它们 ， 那 你 只 能 完成 很 有 限 
的 工作 。 正 因为 如 此 ， 本 书 专门 在 此 介绍 了 异常 一 -对 于 许多 类 库 (例如 提 到 过 的 IO 库 )， 如 果 
不 处 理 异常 ， 你 就 无 法 使 用 它们 。 
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异常 处 理 的 优点 之 一 就 是 它 使 得 你 可 以 在 某 处 集中 精力 处 理 你 要 解决 的 问题 ， 而 在 另 一 处 
处 理 你 编写 的 这 段 代码 中 产生 的 错误 。 尽 管 异常 通常 被 认为 是 一 种 工具 ， 使 得 你 可 以 在 运行 时 
报告 错误 并 从 错误 中 恢复 ， 但 是 我 一 直 怀疑 到 底 有 多 少时 候 “ 恢 复 ” 真 正 得 以 实现 了 ， 或 者 能 
够 得 以 实现 。 我 认为 这 种 情况 少 于 10%， 并 且 即 便 是 这 10%， 也 只 是 将 栈 展开 到 某 个 已 知 的 稳 
定 状 态 ， 而 并 没有 实际 执行 任何 种 类 的 恢复 性 行为 。 无 论 这 是 否 正确 ， 我 一 直 相 信 “ 报 告 ” 功 
能 是 异常 的 精 八 所在。Java 坚 定 地 强调 将 所 有 的 错误 都 以 异常 形式 报告 的 这 一 事实 ， 正 是 它 远 
远 超过 诸如 C++ 这 类 语言 的 长 处 之 一 ， 因 为 在 C++ 这 类 语言 中 ， 需 要 以 大 量 不 同 的 方式 来 报告 
错误 ， 或 者 根本 就 没有 提供 错误 报告 功能 。 一 致 的 错误 报告 系统 意味 着 ， 你 再 也 不 必 对 所 写 的 
每 一 段 代 码 ， 都 质问 自己 “错误 是 否 正在 成 为 漏网 之 鱼 ? ”( 只 要 你 没有 “吞咽 ”异常 ， 这 是 
关键 所 在 ! )。 

就 像 你 将 要 在 后 续 章 节 中 看 到 的 ， 通 过 将 这 个 问题 忆 给 其 他 代码 一 即使 你 是 通过 抛 出 
RuntimeException 来 实现 这 一 点 的 一 你 在 设计 和 实现 时 ， 便 可 以 专注 于 更 加 有 趣 和 富有 挑战 
性 的 问题 了 。 

所 选 习 题 的 答案 都 可 以 在 名 为 The Thinking in Java Annotated Solution Guide 的 电子 文档 中 找 
到 ， 读 者 可 以 从 www.MindView.net 处 购买 此 文档 。 
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可 以 证 明 ， 字 符 串 操作 是 计算 机 程序 设计 中 最 常见 的 行为 。 
尤其 是 在 Java 大 展 拳 脚 的 Web 系 统 中 更 是 如 此 。 在 本 章 中 ， 我 们 将 深入 学 习 在 Java 语 言 中 应 


用 最 广泛 的 String 类 ， 并 研究 与 之 相关 的 类 及 工具 。 
13.1 不 可 变 String 


String 对 象 是 不 可 变 的 。 查 看 JDK 文 档 你 就 会 发 现 ，String 类 中 每 一 个 看 起 来 会 修改 String 


值 的 方法 ， 实 际 上 都 是 创建 了 一 个 全 新 的 String 对 象 ， 以 包含 修改 后 的 字符 串 内容 。 而 最 初 的 
String 对 象 则 丝毫 未 动 。 


看 看 下 面 的 代码 ; 


//: strings/Immutable. java 
import static net.mindview.util.Print.*; 


public class Immutable { 
public static String upcase(String $)'{ 
return s.toUpperCase(); 


} 
public static void main(String[] args) { 
String q = "howdy"; 
print(q); // howdy 
String qq = upcase(q); 
print(qq); // HOWDY 
print(q); // howdy 


} 
} /* Output: 
howdy 
HOWDY 
howdy 
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当 把 q 传 给 upease() 方 法 时 ， 实 际 传递 的 是 引用 的 一 个 拷贝 。 其 实 ， 每 当 把 String 对 象 作为 方法 
的 参数 时 ， 都 会 复制 一 份 引用 ， 而 该 引用 所 指 的 对 象 其 实 一 直 待 在 单一 的 物理 位 置 上 ， 从 未 


回 到 upease0 的 定义 ， 传 人 其 中 的 引用 有 了 名 字 s， 只 有 upease0 运 行 的 时 候 ， 局 部 引用 s 才 


存在 。 一 旦 upease0 运 行 结束 ，s 就 消失 了 。 当 然 了 ，upease0 的 返回 值 ， 其 实 只 是 最 终结 果 的 引 
用 。 这 足以 说 明 ，upcase0 返 回 的 引用 已 经 指向 了 一 个 新 的 对 象 ， 而 原本 的 q 则 还 在 原 地 。 


String 的 这 种 行为 方式 其 实 正 是 我 们 想 要 的 。 例 如 : 


String s = "asdf"; 
String x = Immutable.upcase(s); 


难道 你 真 的 希望 upease0 改 变 其 参数 吗 ? 对 于 一 个 方法 而 言 ， 参 数 是 为 该 方法 提供 信息 的 ， 而 不 
是 想 让 该 方法 改变 自己 的 。 在 阅读 这 段 代码 时 ， 读 者 自然 就 会 有 这 样 的 感觉 。 这 一 点 很 重要 ， 
正 是 有 了 这 种 保障 ， 才 使 得 代码 易于 编写 与 阅读 。 


13.2 重 载 “+” 与 StringBuilder 


String 对 象 是 不 可 变 的 ， 你 可 以 给 一 个 String 对 象 加 任意 多 的 别名 。 因 为 String 对 象 具有 
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只 读 特性 ， 所 以 指向 它 的 任何 引用 都 不 可 能 改变 它 的 值 ， 因 此 ， 也 就 不 会 对 其 他 的 引用 有 什 
么 影响 。 

不 可 变性 会 带 来 一 定 的 效率 问题 。 为 String 对 象 重 载 的 “+ ”操作 符 就 是 一 个 例子 。 重 载 的 
意思 是 ， 一 个 操作 符 在 应 用 于 特定 的 类 时 ， 被 赋予 了 特殊 的 意义 (用 于 String 的 “+” 与 “+=” 
是 Java 中 仅 有 的 两 个 重 载 过 的 操作 符 ， 而 Java 并 不 允许 程序 员 重 载 任何 操作 符 。)。 

操作 符 “+” 可 以 用 来 连接 String: 


41: strings/Concatenation. java 


public class Concatenation { 
public static void main(String[] args) { 
String mango = "mango": 
String s = "abc" + mango + "def" + 47; 
System.out.printin(s); 


} A Output: 
abcmengodet47 
Ii~ 

可 以 想象 一 下 ， 这 段 代 码 可 能 是 这 样 工作 的 : String 可 能 有 一 个 append( 方 法 ， 它 会 生成 一 个 新 
的 String 对 象 ， 以 包含 “abc” 与 mango 连 接 后 的 字符 串 。 然 后 ， 该 对 象 再 与 “def” 相 连 ， 生 成 
另 一 个 新 的 String 对 象 ， 依 此 类 推 。 

这 种 工作 方式 当然 也 行 得 通 ， 但 是 为 了 生成 最 终 的 String， 此 方式 会 产生 一 大 堆 需 要 垃圾 回 
收 的 中 间 对 象 。 我 猜想 ，Java 设 计 师 一 开始 就 是 这 么 做 的 〈 这 也 是 软件 设计 中 的 一 个 教训 : BR 
非 你 用 代码 将 系统 实现 ， 并 让 它 动 起 来 ， 否 则 你 无 法 真正 了 解 它 会 有 什么 问题 ) ， 然 后 他 们 发 现 
其 性 能 相当 精 糕 。 

想 看 看 以 上 代码 到 底 是 如 何 工作 的 吗 ， 可 以 用 JDK 自 带 的 工具 javap 来 反 编译 以 上 代码 。 命 
令 如 下 : 

javap -c Concatenation 
这 里 的 -c 标 志 表 示 将 生成 JVM 字 节 码 。 我 别 除 掉 了 不 感 兴趣 的 部 分 ， 然 后 作 了 一 点 点 修改 ， 于 
是 有 了 以 下 的 字 节 码 ， 


public static void main(java.lang.String[]); 
Code: 
Stack=2, Locals=3, Args_size=1 
@: ldc #2; //String mango 


2: astore_1 
3: new #3; //class StringBuilder 

6: dup 

7: invokespecial #4; //StringBuilder."<init>":() 


10: tde #5; //String abc 

12: invokevirtual #6; //StringBuilder append: (String) 
15: aload_1 

16: invokevirtual #6; //StringBuilder. append: (String) 
19: lde #7; //String def 

21: invokevirtual #6; //StringBuilder. append: (String) 
24: bipush 47 

26: invokevirtual #8; //StringBuilder append: (I) 

29: invokevirtual #9; //StringBuilder. toString: O 


O cr+ 克 许 程序 员 任意 重 载 操作 符 。 由 于 这 通常 是 很 复杂 的 过 程 (参见 《C++ 编程 思想 (第 2 版 )》 第 10 章 一 本 
书 中 英文 版 均 已 由 机 械 工业 出 版 社 出 版 )， 所 以 Java 设 计 者 认为 这 是 “ 精 粒 的 ”功能 ， 不 应 该 包括 在 Java 中 。 
其 实 重 载 操 作 符 并 没有 精 粒 到 只 能 让 它们 自己 去 重 载 的 地 步 ， 但 具有 讽刺 意味 的 是 ， 与 C++ 相 比 ， 在 Java 中 使 
用 操作 符 重 载 要 容易 得 多 。 这 一 点 可 以 在 Python (参考 www.Python org) 与 C# 中 看 到 ， 它 们 都 具有 垃圾 回收 与 
简单 易 懂 的 操作 符 重 载 机 制 。 
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32: astore_2 
33: getstatic #10; //Field System.out:PrintStream; 
36: aload_2 
37: invokevirtual #11; // PrintStream.println: (String) 
40: return 
如 果 你 有 汇编 语言 的 经 验 ， 以 上 代码 一 定 看 着 眼熟 ， 其 中 的 dup 与 ivokevirtural 语 句 相 当 于 Java 
虚拟 机 上 的 汇编 语句 。 即 使 你 完全 不 了 解 汇编 语言 也 无 须 担 心 ， 需 要 注意 的 重点 是 ;编译 器 自 
动 引 入 了 java.lang.StringBuilder 类 。 虽 然 我 们 在 源 代 码 中 并 没有 使 用 StringBuilder 类 ， 但 是 编 
译 器 却 自作 主张 地 使 用 了 它 ， 因 为 它 更 高 效 。 
在 这 个 例子 中 ， 编 译 器 创建 了 一 个 StringBuilder 对 象 ， 用 以 构造 最 终 的 String， 并 为 每 个 字 
符 串 调用 一 次 StringBuilder 的 append0 方 法 ， 总 计 四 次 。 最 后 调用 toString0 生 成 结果 ， 并 存 为 s 
(使 用 的 命令 为 astore_2) 。 
现在 ， 也 许 你 会 觉得 可 以 随意 使 用 String 对 象 ， 反 正 编 译 器 会 为 你 自动 地 优化 性 能 。 可 是 在 
这 之 前 ， 让 我 们 更 深入 地 看 看 编译 器 能 为 我 们 优化 到 什么 程度 。 下 面 的 程序 采用 两 种 方式 生成 
一 个 String: 方法 一 使 用 了 多 个 String 对 象 ， 方 法 二 在 代码 中 使 用 了 StringBuilder。 
/1: strings/WhitherStringBuilder. java 
public class WhitherStringBuilder { 
public String implicit(String[] fields) { 
String result = ""; 
for(int i = 6; i < fields.length; i++) 


result += fields[1]; 
return result; 


d 
public String explicit(String[] fields) { 
StringBuilder result = new StringBuilder(); 
for(int 1 = @; i < fields. length; i++) 
result. append(fields[i]); 
return result. toString(); 


} 
} Mi~ 
现在 运行 javap -c WitherStringBuilder， 可 以 看 到 两 个 方法 对 应 的 (简化 过 的 ) 字 节 码 。 首 先是 
implicit( 方 法 ; 
public java.lang.String implicit(java. lang.String[]); 
Code: 
6: tde #2; //String 
:  astore_2 
iconst_@ 
istore_3 
jload_3 
aload_t 
arraylength 


if_icmpge 38 
new #3; //class StringBuilder 

dup 

invokespecial #4; // StringBuilder."<init>":() 
aload_2 

invokevirtual #5; // StringBuilder. append: () 
atoad_1 

jload_3 

aaload 

invokevirtual #5: // StringBuilder. append: () 
invokevirtual #6; // StringBuilder. toString: () 
astore_2 

dine 3, 1 

goto 5 

aload_2 

areturn 
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注意 从 第 8 行 到 第 35 行 构成 了 一 个 循环 体 。 第 8 行 :对 堆栈 中 的 操作 数 进行 “大 于 或 等 于 的 整数 
比较 运算 "， 循 环 结束 时 跳 到 第 38 行 。 第 35 行 : 返回 循环 体 的 起 始点 (第 5 行 )。 要 注意 的 重点 是 : 
StringBuilder 是 在 循环 之 内 构造 的 ， 这 意味 着 每 经 过 循环 一 次 ， 就 会 创建 一 个 新 的 StringBuilder 
对 象 。 


下 面 是 explicit0 方 法 对 应 的 字 节 码 : 
public java.lang.String explicit(java.lang. String[]); 
Code: 
©: new #3; //class StringBuilder 
3: dup 
4: invokespecial #4: // StringBuilder. "<init>":() 
7: astore_2 
8: iconst_@ 
9: istore 3 
16: iload_3 
11: aload_1 


12: arraylength 
13: if_icmpge 30 


16: aload_2 

17: aload_1 

18: iload_3 

19: aaload 

20: invokevirtual #5; // StringBuilder. append: () 
23: pop 

24: dinc 3, 1 

27: goto 10 

30: aload_2 


31: invokevirtual #6; // StringBuilder. toString: () 
34: areturn 


可 以 看 到 ， 不 仅 循环 部 分 的 代码 更 简短 、 更 简单 ， 而 且 它 只 生成 了 一 个 StringBuilder 对 象 。 显 
式 地 创建 StringBuilder 还 允许 你 预先 为 其 指定 大 小 。 如 果 你 已 经 知道 最 终 的 字符 申 大 概 有 多 长 ， 
那 预先 指定 StringBuilder 的 大 小 可 以 避免 多 次 重新 分 配 缓冲 。 

因此 ， 当 你 为 一 个 类 编写 toString( 方 法 时 ， 如 果 字 符 串 操作 比较 简单 ， 那 就 可 以 信赖 编译 
器 ， 它 会 为 你 合理 地 构造 最 终 的 字符 串 结果 。 但 是 ， 如 果 你 要 在 toString0 方 法 中 使 用 循环 ， 那 
么 最 好 自己 创建 一 个 StringBuilder 对 象 ， 用 它 来 构造 最 终 的 结果 。 请 参考 以 下 示例 ， 


/1/: strings/UsingstringBuilder .java 
import java.util.*; 


public class UsingStringBuilder { 
public static Random rand = new Random(47); 
public String toString() { 
StringBuilder result = new StringBuilder ("[*); 
for(int i = 0; i < 25; i++) { 
result. append(rand.nextInt(108)) ; 
result.append(", "); 
$ 
result.delete(result. length()-2, result.length()); 
result.append("]"); 
return result. toString(); 
} 
public static void main(String[] args) { 
UsingStringBuilder usb = new UsingStringBui lder(); 
System. out.printin(usb) ; 
} 
} /* Output: 
(58, 55, 93, 61, 61, 29, 68, 6, 22, 7, 88, 28, 51, 89, 9, 


78, 98, 61, 20, 58, 16, 40, 11, 22, 4) 
Whim 


最 终 的 结果 是 用 append0 语 句 一 点 点 拼接 起 来 的 。 如 果 你 想 走 捷径 ,例如 append(a+“:”+c)， 
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那 编译 器 就 会 掉 人 陷阱 ， 从 而 为 你 另外 创建 一 个 StringBuilder 对 象 处 理 括号 内 的 字符 串 操作 。 
如 果 拿 不 准 该 用 哪 种 方式 ， 随 时 可 以 用 javap 来 分 析 你 的 程序 。 
StringBuilder 提 供 了 丰富 而 全 面 的 方法 ,包括 insertO0、repleace0、substring0 甚 至 reverse0， 
但 是 最 常用 的 还 是 append0 和 toString0。 还 有 delete(0 方 法 ， 上 面 的 例子 中 我 们 用 它 删除 最 后 一 
个 逗号 与 空格 ， 以 便 添加 右 括号 。 
StringBuilder 是 Java SE5 引 入 的 ， 在 这 之 前 Java 用 的 是 StringBuffer。 后 者 是 线程 安全 的 
(参见 第 21 章 ) ， 因 此 开销 也 会 大 些 ， 所 以 在 Java SE5/6 中 ， 字 符 串 操作 应 该 还 会 更 快 一 点 。 
练习 1，(2) 分 析 reusing/SprinklerSystem.java 的 SprinklerSystem.tString0 方 法 ， 看 看 明确 地 
使 用 StringBuilder 是 否 确实 能 够 避免 产生 过 多 的 StringBuilder 对 象 。 


13.3 无 意识 的 递归 


Java 中 的 每 个 类 从 根本 上 都 是 继承 自 Object， 标 准 容器 类 自然 也 不 例外 。 因 此 容器 类 都 有 
toString0 方 法 ， 并 且 覆 写 了 该 方法 ， 使 得 它 生成 的 String 结 果 能 够 表达 容器 自身 ， 以 及 容器 所 
包含 的 对 象 。 例 如 ArrayList.toString0， 它 会 遍历 ArrayList 中 包含 的 所 有 对 象 ， 调 用 每 个 元 素 
上 的 toString0 方 法 : 


//: strings/ArrayListDisplay. java 

import generics.coffee.*; 

import java.util. *; 

public class ArrayListDisplay { ‘ 

public static void main(String[] args) { 
ArrayList<Coffee> coffees = new Arraylist<Coffee>(); 
for (Coffee c : new CoffeeGenerator (10)) 
coffees. add(c) ; 

System.out.printin(coffees) ; 


) /* Output: 
(Americano ©, Latte 1, Americano 2, Mocha 3, Mocha 4, Breve 
5, Americano 6, Latte 7, Cappuccino 8, Cappuccino 9) 

Wh 


如 果 你 希望 toString0 方 法 打印 出 对 象 的 内 存 地 址 ， 也 许 你 会 考虑 使 用 this 关 键 字 ; 


1/: strings/InfiniteRecursion. java 
// Accidental recursion. 

// {RunByHand) 

import java.util.*; 


public class InfiniteRecursion { 
public String toString() { 
return " InfiniteRecursion address: " + this + "\n"; 


} 
public static void main(String(] args) { 
List<InfiniteRecursion> v = 
new ArrayList<InfiniteRecurston>(); 
for(int 1 = 0; i < 10; i++) 
v.add(new InfiniteRecursion()); 
System.out.printin(v); 


} 
} /LA 
当 你 创建 了 InfiniteRecursion 对 象 ， 并 将 其 打印 出 来 的 时 候 ， 你 会 得 到 一 串 非 常 长 的 异常 。 
如 果 你 将 该 InfiniteRecursion 对 象 存 人 一 个 ArrayList 中 ， 然 后 打印 该 ArrayList， 你 也 会 得 到 同 
样 的 异常 。 其 实 ， 当 如 下 代码 运行 时 


“InfiniteRecursion address: " + this 
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这 里 发 生 了 自动 类 型 转换 ， 由 InfiniteRecursion 类 型 转换 成 String 类 型 。 





为 编译 器 看 到 一 


个 String 对 象 后 面 跟着 一 个 “+”， 而 再 后 面 的 对 象 不 是 String， 于 是 编译 器 试 着 将 this 转 换 成 一 
个 String。 它 怎么 转换 呢 ， 正 是 通过 调用 this 上 的 toString0 方 法 ， 于 是 就 发 生 了 递归 调用 。 

如 果 你 真 的 想 要 打印 出 对 象 的 内 存 地址， 应 该 调用 ObjecttoString0 方 法 ， 这 才 是 负责 此 任 
务 的 方法 。 所 以 ， 你 不 该 使 用 this， 而 是 应 该 调用 sapertoString0 方 法 。 

练习 2: (1) 修复 InfiniteReeursionjava。 


13.4 String 上 的 操作 
以 下 是 String 对 象 具备 的 一 些 基本 方法 。 重 载 的 方法 归纳 在 同一 行 中 : 














方 法 参数 ， 重 载 版 本 应 用 
构造 器 TREE: 默认 版 本 ，String，String- 创建 String 对 象 
Builder，StringBuffer，char 数 组 ，byte 
| 数组 
length) String 中 字符 的 个 数 
charAtQ) mI gi 取得 String 中 该 索引 位 置 上 的 char 
getChars(), getBytes() 要 复制 部 分 的 起 点 和 终点 复制 char 或 byte 到 一 个 目标 数组 中 





的 目标 数组 ， 目 标 数组 的 起 始 索引 





toCharArray0 


生成 一 个 char[]， 人 包含 String 的 所 有 字符 





equals(), equalsignoreCase() 


与 之 进行 比较 的 String 


比较 两 个 String 的 内 容 是 否 相同 























compareToO) 与 之 进行 比较 的 String 按 词典 顺序 比较 String 的 内 容 ， 比 较 结 
果 为 负数 、 零 或 正 数 。 注 意 ， 大 小 写 并 
不 等 价 
contains() 要 搜索 的 CharSequence 如 果 该 String 对 象 包含 参数 的 内 容 ， 则 
返回 true ， 
contentEquals() 与 之 进行 比较 的 CharSequence 或 String- 如 果 该 String 与 参数 的 内 容 完全 一 致 ， 
Buffer 则 返回 true 
equalsIgnoreCase() 与 之 进行 比较 的 String 忽略 大 小 写 ， 如 果 两 个 String 的 内 容 相 
同 ， 则 返回 true 
regionMatcher() 该 String 的 索引 偏 移 量 ， 另 一 个 String 返回 boolean 结 果 ， 以 表明 所 比较 区 域 
及 其 索引 偏 移 量 ， 要 比较 的 长 度 。 重 载 版 | 是 否 相等 
本 增加 了 “忽略 大 小 写 ” 功 能 
startsWith() 可 能 的 起 始 String。 重 载 版 本 在 参数 中 返回 boolean 结 果 ， 以 表明 该 String 是 否 
增加 了 偏 移 胎 以 此 参数 起 始 
endsWith() 该 String 可 能 的 后 组 String 返回 boolean 结 果 ， 以 表明 此 参数 是 否 


该 字符 囊 的 后 级 





indexOf0, lastIndexOfO 


String，String 与 起 始 索引 


le 
重 载 版 本 包括 : char，char 与 起 始 索引 , 


如 果 该 String 并 不 包含 此 参数 ， 就 返回 
一 1， 否则 返回 此 参数 在 String 中 的 起 始 索 
1。lastindexOfO 是 从 后 向 前 搜索 





substring() (subSequence()) 





坐标 





ži 
重 载 版 本 : 起 始 索引 ， 起 始 索引 + 终点 


返回 一 个 新 的 String， 以 包含 参数 指定 


的 子 字符 囊 
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( 续 ) 
方 法 参数 ， 重 载 版 本 应 用 
eoncat0 要 连接 的 String 返回 一 个 新 的 String 对 象 ， 内 容 为 原始 
String 连 接 上 参数 String 
replace) BARE, WTS | ”返回 替换 字符 后 的 新 String 对 象 。 如 果 
符 。 也 可 以 用 一 个 CharSequenee 来 替换 另 | 没有 替换 发 生 ， 则 返回 原始 的 String 对 象 
一 个 CharSequence 
toLowerCase() toUpperCase() 将 字符 的 大 小 写 改 变 后 ， 返 回 一 个 新 
String 对 象 。 如 果 没有 改变 发 生 ， 则 返回 
原始 的 String 对 象 
trim 将 String 两 端的 空白 字符 副 除 后 ， 返 回 


一 个 新 的 String 对 象 。 如 果 没有 改变 发 生 ， 
则 返回 原始 的 String 对 象 


valueOf0 KHE: Object, chari], chari), a 返回 一 个 表示 参数 内 容 的 String 
Bik, SEHR, boolean, char, int, 
| tongs Moat, double A 

intern0 为 每 个 唯一 的 字符 序列 生成 一 个 且 仅 
生成 一 个 String 引 用 

从 这 个 表 中 可 以 看 出 ， 当 需要 改变 字符 串 的 内 容 时 ，String 类 的 方法 都 会 返回 一 个 新 的 
String 对 象 。 同 时 ， 如 果 内 容 没 有 发 生 改变 ，String 的 方法 只 是 返回 指向 原 对 象 的 引用 而 已 。 这 
可 以 节约 存储 空间 以 及 避免 额外 的 开销 。 

本 章 稍 后 还 将 介绍 正则 表达 式 在 String 方 法 中 的 应 用 。 


13.5 格式 化 输出 


在 长 久 的 等 待 之 后 ，Java SE5 终 于 推出 了 C 语 言 中 printf0 风 格 的 格式 化 输出 这 一 功能 。 这 不 仅 
使 得 控制 输出 的 代码 更 加 简单 ， 同 时 也 给 与 Java 开 发 者 对 于 输出 格式 与 排列 更 强大 的 控制 能 力 。 。 
13.5.1 printf() 

C 语 言 中 的 printfO 并 不 能 像 Java 那 样 连接 字符 串 ， 它 使 用 一 个 简单 的 格式 化 字符 串 ， 加 上 要 
插入 其 中 的 值 ， 然 后 将 其 格式 化 输出 。printf0 并 不 使 用 重 载 的 “+” 操 作 符 (CARR) 来 连 
接 引号 内 的 字符 串 或 字符 串 变 量 ， 而 是 使 用 特殊 的 占 位 符 来 表示 数据 将 来 的 位 置 。 而 且 它 还 将 
插入 格式 化 字符 串 的 参数 ， 以 逗号 分 隔 ， 排 成 一 行 。 

例如 : 

printf ("Row 1: [%d %f]\n", x, y); 
这 一 行 代码 在 运行 的 时 候 ， 首 先 将 x 的 值 插入 到 %d 的 位 置 ， 然 后 将 y 的 值 插 入 到 %f 的 位 置 。 这 
些 占 位 符 称 作 格 式 修饰 罕 ， 它 们 不 但 说 明了 插入 数据 的 位 置 ， 同 时 还 说 明了 将 插入 什么 类 型 的 
变量 ,以 及 如 何 对 其 格式 化 。 在 这 个 例子 中 ，%d 表 示 x 是 一 个 整数 ，%f 表 示 y 是 以 一 个 浮 点 数 
(float 或 者 double) 。 
13.5.2 System.out.format() 

Java SE53 引 入 的 format 方 法 可 用 于 PrintStream 或 PrintWriter 对 象 (我 们 将 在 第 18 章 学 习 它 
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们 )， 其 中 也 包括 System.out 对 象 。format0 方 法 模仿 自 C 的 printf0。 如 果 你 比较 怀旧 的 话 ， 也 可 
以 使 用 printf0。 以 下 是 一 个 简单 的 示例 : 


//: strings/SimpleFormat. java 


public class SimpleFormat { 

public static void main(Stringl] args) { 

int x = 5; 

double y = 5.332542; 

// The old way: 

System.out.printin("Row 1: ("+x +" " +y #"]"); 
` 77 The new way: 

System.out.format("Row 1: [%d %f]\n", x, y): 

1 or 

System.out.printf("Row 1: (%d %f]\n", x, y); 


} 
} /* Output: 
Row 1: [5 5.332542] 
Row 1: [5 5.332542] 
Row 1: {5 5.332542) 
Whim 


可 以 看 到 ，format0 与 printf0 是 等 价 的 ， 它 们 只 需要 一 个 简单 的 格式 化 字符 串 ， 加 上 一 串 参数 
即 可 ， 每 个 参数 对 应 一 个 格式 修饰 符 。 
13.5.3 Formatter 类 

在 Java 中 ， 所 有 新 的 格式 化 功能 都 由 java.utilFormatter 类 处 理 。 可 以 将 Formatter 看 作 一 个 
翻译 器 ， 它 将 你 的 格式 化 字符 串 与 数据 翻译 成 需要 的 结果 。 当 你 创建 一 个 Formatter 对 象 的 时 候 ， 
需要 向 其 构造 器 传递 一 些 信息 ， 告 诉 它 最 终 的 结果 将 向 哪里 输出 ; 


//: strings/Turtle. java 
import java.io.* 
import java.util.*; 








public class Turtle { 
private String name; 
private Formatter f; 
public Turtle(String name, Formatter f) { 
this.name = name; 
this.f = f; 


public void move(int x, int y) { 
f.format("%s The Turtle is at (%d,%d)\n", name, x, y); 
} 
public static void main(String[] args) { 
PrintStream outAlias = System.out; 
Turtle tommy = new Turtle(*Tommy", 
new Formatter(System.out)) ; 
Turtle terry = new Turtle("Terry", 
new Formatter(outAlias)); 
tommy move (9,8) 
terry.move(4,8 
tommy .move (3.4 
terry.move(2,5); 
tommy „move (3,3): 
terry.move (3,3); 





} 

} /* Output: 

Tommy The Turtle is at (0,0) 
Terry The Turtle is at (4,8) 
Tommy The Turtle is at (3,4) 
Terry The Turtle is at (2,5) 
Tommy The Turtle is at (3,3) 
Terry The Turtle is at (3,3) 
AL] 
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所 有 的 tommy 都 将 输出 到 System.out， 而 所 有 的 terry 则 都 输出 到 System.out 的 一 个 别名 中 。 
Formatter 的 构造 器 经 过 重 载 可 以 接受 多 种 输出 目的 地 ， 不 过 最 常用 的 还 是 PrintStream0 (如 上 
例 )、OutputStream 和 File。 在 第 18 章 中 你 将 看 到 与 之 相关 的 更 多 信息 。 

练习 3: (1) 修改 Turtle.java， 使 之 将 结果 输出 到 System.err 中 。 

前 面 的 示例 中 还 使 用 了 一 个 新 的 格式 化 修饰 符 %s， 它 表示 插入 的 参数 是 String 类 型 。 这 个 
例子 使 用 的 是 最 简单 类 型 的 格式 修饰 符 一 一 它 只 具有 转换 类 型 而 没有 其 他 功能 。 


13.5.4 格式 化 说 明 符 

在 插入 数据 时 ， 如 果 想 要 控制 空格 与 对 齐 ， 你 需要 更 精细 复杂 的 格式 修饰 符 。 以 下 是 其 抽 
象 的 语法 ， 

%largument_index$] [flags] [width] [.precision] conversion 

最 常见 的 应 用 是 控制 一 个 域 的 最 小 尺寸 ， 这 可 以 通过 指定 width 来 实现 。Formatter 对 象 通 
过 在 必要 时 添加 空格 ， 来 确保 一 个 域 至 少 达到 某 个 长 度 。 在 默认 的 情况 下 ， 数 据 是 右 对 齐 ， 不 
过 可 以 通过 使 用 “-” 标 志 来 改变 对 齐 方向 。 

与 width 相对 的 是 precision， 它 用 来 指明 最 大 尺寸 。width 可 以 应 用 于 各 种 类 型 的 数据 转换 ， 
并 且 其 行为 方式 都 一 样 。precision 则 不 然 ， 不 是 所 有 类 型 的 数据 都 能 使 用 precision， 而 且 ， 应 用 
于 不 同类 型 的 数据 转换 时 ，precision 的 意义 也 不 同 。 在 将 precision 应 用 于 String 时 ， 它 表示 打印 
String 时 输出 字符 的 最 大 数量 。 而 在 将 precision 应 用 于 浮 点 数 时 ， 它 表示 小 数 部 分 要 显示 出 来 的 
位 数 (默认 是 6 位 小 数 ) ， 如 果 小 数位 数 过 多 则 伟人 ， 太 少 则 在 尾部 补 零 。 由 于 整数 没有 小 数 部 
分 ， 所 以 precision 无 法 应 用 于 整数 ， 如 果 你 对 整数 应 用 precision， 则 会 触发 异常 。 

下 面 的 程序 应 用 格式 修饰 符 来 打印 一 个 购物 收据 : 


//: strings/Receipt.java 
import java.util.*; 


public class Receipt { 

Private double total = 0; 

private Formatter f = new Formatter(System.out) 
public void printTitle() { 

f. format ("%-15s %5s %10s\n", 

f. format ("%-15s %5s %10s\n", 






25 


} 

public void print(String name, int qty, double price) { 
f. format ("%-15.15s %5d %10.2f\n", name, qty, price); 
total += price; 

} 

public void printTotal() { 
f.format("%-15s %5s %10.2f' 
f.format("%-15s %5s %10s\n 
f.format("%-15s %5s %10.2f\n", 

total * 1.06); 


total*0.06); 
0; 








tal", 


} 

public static void main(String{] args) { 
Receipt receipt = new Receipt(): 
receipt.printTitle(): 
receipt.print("Jack's Magic Beans", 4, 4.25); 
receipt.print("Princess Peas", 3, 5.1); 
receipt.print("Three Bears Porridge”. 1, 14.29); 
receipt.printTotal(); 


} 
} 7* Output: 
Item Qty Price 
Jack's Magic Be 4 4.25 


Princess Peas 3 5.10 
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Three Bears Por 1 14.29 
Tax 1.42 
Total 25.06 
Wim 


正如 你 所 见 ， 通 过 相当 简洁 的 语法 ，Formatter 提 供 了 对 空格 与 对 齐 的 强大 控制 能 力 。 在 该 程序 
中 ， 为 了 恰当 地 控制 间隔 ， 格 式 化 字符 串 被 重复 地 利用 了 多 遍 。 

练习 4: (3) 修改 Receiptjava， 令 所 有 的 宽度 都 由 一 个 常量 来 控制 。 目 的 是 使 宽度 的 改变 更 
容易 ， 只 需 修改 一 处 的 值 即 可 。 
13.5.5 Formatter 转 换 












下 面 的 表格 包含 了 最 常用 的 类 型 转换 : 
—— 
类 型 转换 字符 
d 整数 型 (十进制 ) e 浮 点 数 (科学 计数 ) 
© Unicode 字 符 x 整数 《十 六 进 制 ) 
b Boolean 值 h 散 列 码 (十 六 进 制 ) 
s String % 字符 “9%" 
f 浮 点 数 《十进制 ) 





下 面 的 程序 演示 了 这 些 转换 是 如 何 工作 的 : 


//: strings/Conversion. java 


import java.math 
import java.uti 
public class Conversion { 
public static void main(String[] args) { 
Formatter f = new Formatter(System.out) ; 








char u = 网 | 
System.out.printin("u = ‘a'"); 
f.format("s: %s\n", u): 

17 fformat("d: %d\n", u): 
f.format("c: %c\n", u); 
f.format("b: %b\n", u); 

17 f.format("t: KF\N", u); 

// f.format("e: %e\n", u); 

11 t.tormat("x: %x\n", u); 
f.format("h: %h\n", u); 









int v = 121; 
System.out.printin("v = 121"); 
f.format("d: %d\n", v); 
f.format("c: %c\n", v); 
f.format("b: %b\n", v); 
f.format("s: %s\n", v); 
‘%F\N". v); 

: %e\n", v); 
f.format("x: %x\n", v); 
f.format("h: %h\n", v): 





Biginteger w = new BigInteger ("50000000000000") ; 
System. out. printin( 

"w = new BigInteger (\"50980000000000\")"); 
f.format("d: %d\n", w); 
// f.format("c: 
f.format( 







// f.format("e: %e\n", 
f.format("x: &x\n", w): 
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f.format("h: %h\n", w); 


double x = 179.543; 
System.out.printin("x 
/1/ £.format("d: %d\n", x: 
11 £.format("c: %e\n", x); 
f.format("b: %b\n", x); 
f.format("s: %s\n", x); [519] 
f.format("f: %f\n", x); 
f.format("e: %e\n", x); 
11 €.format("x: %x\n", x); 
f.format("h: %h\n", x); 


179.543"); 




















Conversion y = new Conversion(); 
System.out.println("y = new Conversion()"); 
71 f.format("d: %d\n" 
// £.format("c: %c\n" 
f.format("b: %b\n", y) 
f.format("s: %s\n", y); 
44 f.format("f: %f\n", y); 
// f.format("e: %e\n". y 
/1 f.format(*x: %x\n", y); 
f.format("h: %h\n", y); 













boolean z = false; 
System.out.printin("z = false"); 
77 f.format("d: %d\n", z 
1/ f.format("c: %e\n", 2); 
f.format("b: %b\n", 2): 
f.format("s: %s\n", z) 
11 f.format("f: %f\n" 
11 f.format("e: %e\n", 
17 t.tormat("x: %x\n" 
f.format("h: &h\n", z); 

} 

7* Output: (Sample) 

faites 













121 
79 

79 

w = new BigInteger ("50000000000000") 
50000000000000 
true 

: 50000000000000 
247988342000 
8842a1a7 

179.543 

: true 

: 179.543 

: 179.543000 

: 1.795430e+02 

: lef462c 

= new Conversion() 
: true 

: Conversion@9cab16 
: 9cab16 

= false 

: false 

false 








x 
b 
s 
f 
e 





woNnzuce 
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h: 4d5 
Whim 


被 注释 的 代码 表示 ， 针 对 相应 类 型 的 变量 ， 这 些 转换 是 无 效 的 。 如 果 执行 这 些 转换 ， 则 会 触发 
异常 。 

注意 ， 程 序 中 的 每 个 变量 都 用 到 了 b 转 换 。 虽 然 它 对 各 种 类 型 都 是 合法 的 ， 但 其 行为 却 不 一 
定 与 你 想象 的 一 致 。 对 于 boolean 基 本 类 型 或 Boolean 对 象 ， 其 转换 结果 是 对 应 的 true 或 false。 但 
是 ， 对 其 他 类 型 的 参数 ， 只 要 该 参数 不 为 null， 那 转换 的 结果 就 永远 都 是 true。 即 使 是 数字 0， 
转换 结果 依然 为 true， 而 这 在 其 他 语言 中 (包括 C) ， 往 往 转换 为 false。 所 以 ， 将 b 应 用 于 非 布尔 
类 型 的 对 象 时 请 格外 小 心 。 

还 有 许多 不 常用 的 类 型 转换 与 格式 修饰 符 选 项 ， 你 可 以 在 JDK 文 档 中 的 Formatter 类 部 分 找 
到 它们 。 

练习 5: (5) 针对 前 边 表格 中 的 各 种 基本 转化 类 型 ， 请 利用 所 有 可 能 的 格式 修饰 符 ， 写 出 一 
个 尽 可 能 复杂 的 格式 化 表达 式 。 
13.5.6 String.format() 

Java SE5 也 参考 了 C 中 的 sprintf0 方 法 ， 以 生成 格式 化 的 String 对 象 。String.format0 是 一 个 
static 方 法 ， 它 接受 与 Formatter.format0 方 法 一 样 的 参数 ， 但 返回 一 个 String 对 象 。 当 你 只 需 使 
用 format0 方 法 一 次 的 时 候 ，String.format0 用 起 来 很 方便 。 例 如 : 


//: strings/DatabaseException. java 


public class DatabaseException extends Exception { 
Public DatabaseException(int transactionID, int queryID, 
String message) { 
super (String. format("(t&d, q%d) %s", transactionID, 
queryID, message); 


public static void main(String(] args) { 
try { 
throw new DatabaseException(3, 7, "Write failed"); 
} catch(Exception e) { 
System.out.printin(e); 
} 
} 
} /* Output: 
DatabaseException: (t3, q7) Write failed 
Wham 


其 实在 String.format0 内 部 ， 它 也 是 创建 一 个 Formatter 对 象 ， 然 后 将 你 传人 的 参数 转 给 该 
Formatter。 不 过 ， 与 其 自己 做 这 些 事情 ， 不 如 使 用 便捷 的 String.format(0 方 法 ， 何 况 这 样 的 代 
码 更 清晰 易 读 。 

一 个 十 六 进 制 转 储 (dump) 工具 

在 处 理 二 进 制 文件 时 ， 我 们 经 常 希望 以 十 六 进 制 的 格式 看 看 其 内 容 。 现 在 ， 我 们 就 将 它 作 
为 第 二 个 例子 。 下 面 的 小 工具 使 用 了 String.format0 方 法 ， 以 可 读 的 十 六 进 制 格式 将 字 节 数 组 打 
印 出 来 : 


//: net/mindview/util/Hex. java 
package net.mindview.util; 
import java.io.*; 


public class Hex { 
public static String format (byte[] data) { 
StringBuilder result = new StringBuilder(): 
int n = 0; 
for(byte b : data) { 
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if(n % 16 == 6) 
result. append (String. format ("%@5X: ", n)); 
result.append(String.format("%82X ", b)); 
net; 
if(n % 16 == @) result.append("\n"); 
} 
result.append("\n"); 
return result.toString(); 
sd 
Public static void main(String[] args) throws Exception { 
if(args.length == 6) 
// Test by displaying this class file: 
System.out.println( 
format (BinaryFile.read("Hex.class"))); 
else 
‘System.out .printin( 
format (BinaryFile.read(new File(args{@})))); 


t 

} /* Output: (Sample) 

90000: CA FE BA BE 09 00 09 31 00 52 6A 60 05 09 
00010: 60 23 GA 99 G2 80 22 Ə8 ƏƏ 24 07 0Ə 25 OA 
00020: 09 27 GA 0Ə 28 GO 29 GA 69 G2 GO 2A 68 GO 2B 6A 
00030: 09 2C 99 2D 68 00 2E GA 99 02 69 2F 69 GO 
00040: 31 08 09 32 OA GO 33 60 34 GA GO 15 00 35 
09050: 36 09 37 67 09 38 GA GƏ 12 GO 39 6A 68 33 


“i~ 

为 了 打开 以 及 读 入 二 进 制 文件 ， 我 们 用 到 了 另 一 个 工具 net.mindview.util.BinaryFile， 在 第 
18 章 中 我 会 详细 地 介绍 它 。 这 里 的 read0 方 法 将 整个 文件 以 byte 数 组 的 形式 返回 。 

练习 6，(2) 创建 一 个 包含 int、long、float 与 double 元 素 的 类 。 应 用 String.format0 方 法 为 这 
个 类 编写 toString0 方 法 ， 并 证 明 它 能 正确 工作 。 


13.6 正则 表达 式 


很 久之 前 ， 正 则 表达 式 就 已 经 整合 到 标准 Unix 工 具 集 之 中 ， 例 如 sed 和 awk， 以 及 程序 设计 
语言 之 中 了 ， 例 如 Python 和 Perl (有 些 人 认为 正 是 正则 表达 式 促成 了 Perl 的 成 功 ) 。 而 在 Java 中 ， 
字符 串 操 作 还 主要 集中 于 String、StringBuffer 和 StringTokenizer 类 。 与 正则 表达 式 相 比较 ， 它 
们 只 能 提供 相当 简单 的 功能 。 

正则 表达 式 是 一 种 强大 而 灵活 的 文本 处 理工 具 。 使 用 正则 表达 式 ， 我 们 能 够 以 编程 的 方式 ， 
构造 复杂 的 文本 模式 ， 并 对 输入 的 字符 串 进行 搜索 。 一 旦 找到 了 匹配 这 些 模式 的 部 分 ， 你 就 能 
够 随心 所 欲 地 对 它们 进行 处 理 。 初 学 正则 表达 式 时 ， 其 语法 是 一 个 难点 ， 但 它 确实 是 一 种 简洁 、 
动态 的 语言 。 正 则 表达 式 提供 了 一 种 完全 通用 的 方式 ， 能 够 解决 各 种 字符 串 处 理 相关 的 问题 : 
匹配 、 选 择 、 编 辑 以 及 验证 。 i 
13.6.1 基础 

一 般 来 说 ， 正 则 表达 式 就 是 以 某 种 方式 来 描述 字符 串 ， 因 此 你 可 以 说 ;:“ 如 果 一 个 字符 串 含 
有 这 些 东 西 ， 那 么 它 就 是 我 正在 找 的 东西 .” 例 如 ， 要 找 一 个 数字 ， 它 可 能 有 一 个 负 号 在 最 前 面 ， 
那 你 就 写 一 个 负 号 加 上 一 个 问号 ， 就 像 这 样 : 

要 描述 一 个 整数 ， 你 可 以 说 它 有 一 位 或 多 位 阿拉 伯 数 字 。 在 正则 表达 式 中 ， 用 \d 表 示 一 位 
数字 。 如 果 在 其 他 语言 中 使 用 过 正则 表达 式 ， 那 你 立刻 就 能 发 现 Java 对 反 斜 线 \ 的 不 同 处 理 。 
在 其 他 语言 中 ，\\ 表 示 “ 我 想 要 在 正则 表达 式 中 插入 一 个 普通 的 (字面 上 的 ) 反 斜 线 ， 请 不 要 
给 它 任何 特殊 的 意义 。” 而 在 Java 中 ，\\ 的 意思 是 “我 要 插入 一 个 正则 表达 式 的 反 斜 线 ， 所 以 其 
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后 的 字符 具有 特殊 的 意义 。” 例 如 ， 如 果 你 想 表示 一 位 数字 ， 那 么 正则 表达 式 应 该 是 Wd。 如 果 
你 想 插入 一 个 普通 的 反 斜 线 ， 则 应 该 这 样 \N\。 不 过 换行 和 制 表 符 之 类 的 东西 只 需 使 用 单反 斜 
线 : \n\t。 

要 表示 “一 个 或 多 个 之 前 的 表达 式 ”"， 应 该 使 用 +。 所 以 ， 如 果 要 表示 “可 能 有 一 个 负 号 ， 
后 面 跟着 一 位 或 多 位 数字 "， 可 以 这 样 : 

=2?\\d+ 

应 用 正则 表达 式 的 最 简单 的 途径 ， 就 是 利用 String 类 内 建 的 功能 。 例 如 ， 你 可 以 检查 一 个 
String 是 否 匹 配 如 上 所 述 的 正则 表达 式 : 

/11: strings/IntegerMatch. java 


public class IntegerMatch { 
public static void main(String[] args) { 
System.out.printin("-1234".matches("-?\\d+")); 
System. out.printin("56 
System.out.printin("+9: 
System. out.printin("+9: 







"matche: \\d+")) 5 
".matches("-?\\d+")); 
smatches("(-|\\+)?\\de")); 





} 
} /* Output: 
true 
true 
false 
true 
Whim 


前 两 个 字符 串 满足 对 应 的 正则 表达 式 ， 匹 本 成功。 第 三 个 字符 串 开 头 有 一 个 + ， 它 也 是 一 个 合法 
的 整数 ， 但 与 对 应 的 正则 表达 式 却 不 匹配 。 因 此 ， 我 们 的 正则 表达 式 应 该 描述 为 ;“ 可 能 以 一 个 
加 号 或 减 号 开头 "。 在 正则 表达 式 中 ， 括 号 有 着 将 表达 式 分 组 的 效果 ， 而 坚 直线 | 则 表示 或 操作 。 
也 就 是 : 

GI)? 
这 个 正则 表达 式 表示 字符 串 的 起 始 字符 可 能 是 一 个 -或 +, 或 二 者 皆 没 有 (因为 后 面 跟着 ?修饰 符 ) 。 
因为 字符 + 在 正则 表达 式 中 有 特殊 的 意义 ， 所 以 必须 使 用 \ 将 其 转 义 ， 使 之 成 为 表达 式 中 的 一 个 
普通 字符 。 

String 类 还 自 带 了 一 个 非常 有 用 的 正则 表达 式 工具 一 split0 方 法 ， 其 功能 是 “将 字符 串 从 
正则 表达 式 匹 配 的 地 方 切 开 。 

//: strings/Splitting.java + 


import java.util.*; 


public class Splitting { 

public static String knights = 
“Then, when you have found the shrubbery, you must " + 
"cut down the mightiest tree in the forest... "+ 
“with... a herring!"; 

public static void split(String regex) { 
System.out.printin( 

Arrays. toString(knights.split(regex))); 


} 

public static void main(String[] args) { 
split(" "); // Doesn't have to contain regex chars 
split("\\W+"); // Non-word characters 
split("n\\W+"); // 'n' followed by non-word characters 


} 
} /* Output: 
[Then,, when, you, have, found, the. shrubbery,. you, must, 
cut, down, the, mightiest, tree, in, the, forest..., 
with..., a, herring!) 
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{Then, when, you, have, found, the, shrubbery, you, must, 

cut, down, the, mightiest, tree, in, the, forest, with, a, 

herring) 

[The, whe, you have found the shrubbery, you must cut dow, 

the mightiest tree i, the forest... with... a herring!) 

+~ 

首先 看 第 一 个 语句 ， 注 意 这 里 用 的 是 普通 的 字符 作为 正则 表达 式 ， 其 中 并 不 包含 任何 特殊 
的 字符 。 因 此 第 一 个 splitO 只 是 按 空格 来 划分 字符 串 。 

第 二 个 和 第 三 个 split0 都 用 到 了 \W， 它 的 意思 是 非 单词 字符 (如果 W 小 写 ，\w， 则 表示 一 
个 单词 字符 )。 通 过 第 二 个 例子 可 以 看 到 ， 它 将 标点 字符 删除 了 。 第 三 个 split0 表 示 “ 字 母 0 后 面 
跟着 一 个 或 多 个 非 单词 字符 .” 可 以 看 到 ， 在 原始 字符 串 中 ， 与 正则 表达 式 匹配 的 部 分 ， 在 最 终 
结果 中 都 不 存在 了 。 

String.split0 还 有 一 个 重 载 的 版 本 ， 它 允许 你 限制 字符 囊 分 割 的 次 数 。 

String 类 自 带 的 最 后 一 个 正则 表达 式 工具 是 “替换 "。 你 可 以 只 替换 正则 表达 式 第 一 个 匹配 
的 子 串 ， 或 是 替换 所 有 匹配 的 地 方 。 


//: strings/Replacing. java 
import static net.mindview.util.Print.*; 


public class Replacing { 
static String s = Splitting.knights; 
public static void main(String[] args) { 
print(s.replaceFirst("f\\w+", "located")); 
print(s.replaceall("shrubbery|tree|herring”,"banana")); 
} 
} /* Output: 
Then, when you have located the shrubbery, you must cut 
down the mightiest tree in the forest... with... a herring! 
Then, when you have found the banana, you must cut down the 
mightiest banana in the forest... with... a banana! 
"Whim 


第 一 个 表达 式 要 匹配 的 是 ， 以 字母 f 开 头 ， 后 面 跟 一 个 或 多 个 字母 (注意 这 里 的 w 是 小 写 的) 。 
并 且 只 替换 掉 第 一 个 匹配 的 部 分 ， 所 以 “found” 被 替换 成 “located"。 

第 二 个 表达 式 要 匹配 的 是 三 个 单词 中 的 任意 一 个 ， 因 为 它们 以 竖 直 线 分 隔 表示 “或 "， 并 且 
替换 所 有 匹配 的 部 分 。 

稍 后 你 会 看 到 ，String 之 外 的 正则 表达 式 还 有 更 强大 的 替换 工具 ， 例 如 ， 可 以 通过 方法 调用 
执行 替换 。 而 且 ， 如 果 正 则 表达 式 不 是 只 使 用 一 次 的 话 ， 非 String 对 象 的 正则 表达 式 明 显 具 备 更 
佳 的 性 能 。 

练习 7，(5) 请 参考 java.util.regex.Pattern 的 文档 ， 编 写 一 个 正则 表达 式 ， 检 查 一 个 句子 是 否 
以 大 写字 母 开 头 ， 以 句号 结尾 。 

练习 8: (2) 将 字符 串 Splitting.knights 在 the 和 you 处 分 割 。 

练习 9，(4) 参考 java.utilregex.Pattern 的 文档 ， 用 下 划 线 替换 Splitting.knights 中 的 所 有 元 音 
字母 。 

13.6.2 创建 正则 表达 式 

我 们 首先 从 正则 表达 式 可 能 存在 的 构造 集中 选取 一 个 很 有 用 的 子 集 ， 以 此 开始 学 习 正则 表 

达 式 。 正 则 表达 式 的 完整 构造 子 列表 ， 请 参考 JDK 文 档 java.util.regex 包 中 的 Pattern 类 。 
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字 符 


指定 字符 B 

十 六 进 制 值 为 oxhh 的 字符 

十 六 进 制 表示 为 oxhhhh 的 Unicode 字 符 
制 表 符 Tab 

换行 符 

回 车 

换 页 

转 义 (Escape) 


当 你 学 会 了 使 用 字符 类 (character classes) 之 后 ， 正 则 表达 式 的 威力 才能 真正 显现 出 来 。 以 
下 是 一 些 创建 字符 类 的 典型 方式 ， 以 及 一 些 预定 义 的 类 ; 





2 te 
3 














字符 类 
. 任意 字符 
[abe] 包含 8、b 和 e 的 任何 字符 (和 alble 作 用 相同 ) 
{abe} 除了 a、b 和 e 之 外 的 任何 字符 (否定 ) 
[a-zA-Z] 从 a 到 z 或 从 A 到 Z 的 任 
[abcfhii] 任意 a、b、e、h、i 和 j 字 符 (与 alblelhl 凡 作用 相同 ) (合并 ) 
{a-z&&(hij]] 任意 bh、i 或 j( 交 ) 
\s 空白 符 (空格 、tab、 换 行 、 换 页 和 回 车 ) 
AS 非 空白 符 〈[^s]) 
a 数字 [0-9] 
VD 非 数字 [^0-9] 
Ww 词 字符 [azA-Z0-9] 
Ww 非 词 字符 [A\w] 





这 里 只 列 出 了 部 分 常用 的 表达 式 ， 你 应 该 将 JDK 文 档 中 java.otil.regex.Pattern 那 一 页 加 入 浏 
览 器 书签 中 ， 以 便 在 需要 的 时 候 方便 查询 。 

















逻辑 操作 符 
XY Y 跟 在 X 后 面 
XIY X 或 Y 
00 dikin (capturing group)。 可 以 在 表达 式 中 用 \i 引 用 第 个 捕获 组 
边界 匹配 符 
一 行 的 起 始 B 非 词 的 边界 
$ 一 行 的 结束 \G 前 一 个 匹配 的 结束 





作为 演示 ， 下 面 的 每 一 个 正则 表达 式 都 能 成 功 匹配 字符 序列 “Rudolph”: 


/1/: strings/Rudolph. java 


public class Rudolph { 
public static void main(String{] args) { 
for(String pattern : new String[]{ "Rudolph", 
“[rR]udolph", "{rR] [aeiou] [a-z]ol.*", "R.** }) 
System.out.printin("Rudolph”.matches(pattern)); 


} 
} /* Output: 
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true. 
true 
true 
true 
Whim 


当然 了 ， 我 们 的 目的 并 不 是 编写 最 难 理解 的 正则 表达 式 ， 而 是 尽量 编写 能 够 完成 任务 的 、 
最 简单 以 及 最 必要 的 正则 表达 式 。 一 旦 真正 开始 使 用 正则 表达 式 了 ， 你 就 会 发 现 ， 在 编写 新 的 
表达 式 之 前 ， 你 通常 会 参考 代码 中 已 经 用 到 的 正则 表达 式 。 
13.6.3 量词 
量词 描述 了 一 个 模式 吸收 输入 文本 的 方式 : 
RED: 量词 总 是 贪 禁 的 ， 除 非 有 其 他 的 选项 被 设置 。 贪 禁 表达 式 会 为 所 有 可 能 的 模式 发 
现 尽 可 能 多 的 匹配 。 导 致 此 问题 的 一 个 典型 理由 就 是 假定 我 们 的 模式 仅 能 匹配 第 一 个 可 能 
的 字符 组 ， 如 果 它 是 贪 禁 的 ， 那 么 它 就 会 继续 往 下 匹配 。 
"勉强 型 ， 用 问号 来 指定 ， 这 个 量词 匹配 满足 模式 所 需 的 最 少 字符 数 。 因 此 也 称 作 懒 情 的 
最 少 匹 配 的 、 非 贪 禁 的 、 或 不 贪 禁 的 。 
。 占 有 型 ， 目前， 这 种 类 型 的 量词 只 有 在 Java 语 言 中 才 可 用 (在 其 他 语言 中 不 可 用 )， 并 且 
也 更 高 级 ， 因 此 我 们 大 概 不 会 立刻 用 到 它 。 当 正则 表达 式 被 应 用 于 字符 串 时 ， 它 会 产生 相 
当 多 的 状态 ， 以 便 在 匹配 失败 时 可 以 回潮 。 而 “占有 的 ”量词 并 不 保存 这 些 中 间 状 态 ， 因 
此 它们 可 以 防止 回潮 。 它 们 常常 用 于 防止 正则 表达 式 失控 ， 因 此 可 以 使 正则 表达 式 执行 起 





来 更 有 效 。 ， 
a eH 勉强 型 a 有 型 如 何 匹 配 
x? x” X+ 一 个 或 零 个 X 
x* x"? xe 零 个 或 多 个 X 
X+ X+? Xe 一 个 或 多 个 X 
X{n} X{n)? Xin} 恰好 n 次 X 
X{n,} X{n,}? X{n,}+ 至 少 np 次 X 
X{n,m) X{n,m)}? X{n,m}+ X 至 少 n 次 ， 且 不 超过 m 次 
应 该 非常 清楚 地 意识 到 ， 表 达 式 X 通 常 必须 要 用 圆 括号 括 起 来 ， 以 便 它 能 够 按照 我 们 期 望 的 
效果 去 执行 。 例 如 : 


abc+ 


看 起 来 它 似乎 应 该 匹配 1 个 或 多 个 abc 序 列 ， 如 果 我 们 把 它 应 用 于 输入 字符 串 abcabcabc， 则 实际 
上 会 获得 3 个 匹配 。 然 而 ， 这 个 表达 式 实际 上 表示 的 是 : 匹配 ab， 后 面 跟随 1 个 或 多 个 c。 要 表明 
匹配 1 个 或 多 个 完整 的 abc 字 符 串 ， 我 们 必须 这 样 表示 : 

(abe)+ 

你 会 发 现 ， 在 使 用 正则 表达 式 时 很 容易 混淆 ， 因 为 它 是 一 种 在 Java 之 上 的 新 语言 。 

CharSequence 

接口 CharSequence 从 CharBuffer、String、StringBuffer、StringBuilder 类 之 中 抽象 出 了 字 
符 序列 的 一 般 化 定义 : 


interface CharSequence { 
charAt(int i); 
length(); 
subSequence(int start, int end); 
toString(); 

} 


RA 
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因此 ， 这 些 类 都 实现 了 该 接口 。 多 数 正则 表达 式 操作 都 接受 CharSequence 类 型 的 参数 。 


13.6.4 Pattern#iMatcher 

一 般 来 说 ， 比 起 功能 有 限 的 String 类 ， 我 们 更 愿意 构造 功能 强大 的 正则 表达 式 对 象 。 只 需 导 
入 java.util.regex 包 ， 然 后 用 static Pattern.compile0 方 法 来 编译 你 的 正则 表达 式 即 可 。 它 会 根据 
你 的 String 类 型 的 正则 表达 式 生成 一 个 Pattern 对 象 。 接 下 来 ， 把 你 想 要 检索 的 字符 串 传人 
了 Pattern 对 象 的 matcher0 方 法 。matcher0 方 法 会 生成 一 个 Matcher 对 象 ， 它 有 很 多 功能 可 用 (可 
以 参考 java.util.regext.Matcher 的 JDK 文 档 )。 例 如 ， 它 的 replaceAlI0 方 法 能 将 所 有 匹配 的 部 分 都 
替换 成 你 传人 的 参数 。、 

作为 第 一 个 示例 ， 下 面 的 类 可 以 用 来 测试 正则 表达 式 ， 看 看 它们 能 否 匹 配 一 个 输入 字符 串 。 
第 一 个 控制 台 参 数 是 将 要 用 来 搜索 匹配 的 输入 字符 串 ， 后 面 的 一 个 或 多 个 参数 都 是 正则 表达 式 ， 
它们 将 被 用 来 在 输入 的 第 一 个 字符 串 中 查找 匹配 。 在 Unix/Linux 上 ， 命 令 行 中 的 正则 表达 式 必须 
用 引号 括 起 。 这 个 程序 在 测试 正则 表达 式 时 很 有 用 ， 特 别 是 当 你 想 验 证 它们 是 否 具备 你 所 期 待 
的 匹配 功能 的 时 候 。 


//: strings/TestRegularExpression. java 
// Allows you to easily try out regular expressions. 

17 {Args: abcabcabcdefabc “abc+" "(abc)+" "(abc){2.}" } 
‘import java.util.regex.*; 

import static net.mindview.util.Print.*; 


public class TestRegularExpression { 
public static void main(String[] args) { 
if(args.length < 2) { 
print ("Usage:\njava TestRegularExpression * + 
“characterSequence regularExpression+"); 
System. exit (0); 


) 

print("Input: \"" + args[9] + "\""); 

for(String arg : args) { 
print("Regular expression: \"" + arg + "\""); 
Pattern p = Pattern. compile(arg); 
Matcher m = p.matcher (args{@]); 
while(m.find()) { 

531 A print("Match \"" + m.group() + "\" at positions "+ 
m.start() + "=" + (m.end() - 1)); 





} 
} 


} 
} /* Output: 
Input: "abcabcabcdefabc" 
Regular expression: "abcabcabcdefabc” 
Match "abcabcabcdefabc”at positions 0-14 
Regular expression; "abc+" 
Match "abc" at positions 6-2 
Match "abc" at positions 3-5 
Match “abc” at positions 6-8 
Match “abc” at positions 12-14 
Regular expression: “(abc)+" 
Match "abcabcabc" at positions 0-8 
Match "abc" at positions 12-14 
Regular expression: "(abc){2,}" 
Match “abcabcabc" at positions 6-8 
Sx 


Pattern 对 象 表示 编译 后 的 正则 表达 式 。 从 这 个 例子 中 可 以 看 到 ， 我 们 使 用 已 编译 的 Pattern 
对 象 上 的 matcher0 方 法 ， 加 上 一 个 输入 字符 串 ， 从 而 共同 构造 了 一 个 Mateher 对 象 。 同 时 ， 
了 Pattern 类 还 提供 了 static 方 法 : 





eae 301 





static boolean matches(String regex, CharSequence input) 
该 方法 用 以 检查 regex 是 否 匹配 整个 CharSequence 类 型 的 pput 参 数 。 编 译 后 的 Pattern 对 象 还 提 
供 了 split0 方 法 ， 它 从 匹配 了 regex 的 地 方 分 割 输入 字符 串 ， 返 回 分 割 后 的 子 字符 串 String 数 组 。 
通过 调用 Pattern.matcher0 方 法 ， 并 传人 一 个 字符 串 参数 ， 我 们 得 到 了 一 个 Matcher 对 象 。 
使 用 Matcher 上 的 方法 ， 我 们 将 能 够 判断 各 种 不 同类 型 的 匹配 是 否 成 功 : 


boolean matches() 
boolean lookingAt() 
boolean find() 

boolean find(int start) 


其 中 的 matches() 方 法 用 来 判断 整个 输入 字符 串 是 否 匹配 正则 表达 式 模式 ， 而 lookingAtO 则 用 来 
判断 该 字符 串 〈 不 必 是 整个 字符 串 ) 的 始 部 分 是 否 能 够 匹配 模式 。 

练习 10: (2) 对 字符 串 Java now has regular expressions 验 证 下 列 正则 表达 式 是 否 能 够 发 现 一 
个 匹配 : 


AJava 

\Breg.* 
n.w\sth(ali)s 
s? 

J 

s+ 

s{4) 

s{1). 

s{0,3} 


练习 11: (2) 试用 正则 表达 式 


(Qi) C^ laeiou]) | (\s+[aeiou] ))\w+? [aeiou] \b 
IERE R Arline ate eight apples and one orange while Anita hadn’t any, 


"Arline ate eight apples and one orange while Anita hadn't 
any" 


find() 
Matcherfind( 方 法 可 用 来 在 CharSequence 中 查找 多 个 匹配 。 例 如 


//: strings/Finding. java 
import java.util.regex.*; 
import static net.mindview.util.Print.*; 


public class Finding { 
public static void main(String{] args) { 
Matcher m = Pattern.compile("\\w+") 
„matcher ("Evening is full of the linnet's wings"); 
while(m.find()) 
printnb(m.group() + " "); 
print(); 
int i = 0; 
while(m.find(i)) { 
printnb(m.group() + * "); 
i++; 
}》 
$ 
} /* Output: 
Evening is full of the linnet s wings 


Evening vening ening ning ing ng g is is s full full ull 11 
1 of of f the the he e linnet linnet innet nnet net et t s 
s Wings wings ings ngs gs s 

Whim 


模式 \w+ 将 字符 串 划 分 为 单词 。find0 像 迭代 器 那样 前 向 遍历 输入 字符 串 。 而 第 二 个 find0 能 
够 接收 一 个 整数 作为 参数 ， 该 整数 表示 字符 串 中 字符 的 位 置 ， 并 以 其 作为 搜索 的 起 点 。 从 结果 
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中 可 以 看 出 ， 后 一 个 版 本 的 find0 方 法 能 根据 其 参数 的 值 ， 不 断 重新 设 定 搜索 的 起 始 位 置 。 

组 (Groups) 

组 是 用 括号 划分 的 正则 表达 式 , 可 以 根据 组 的 编号 来 引用 某 个 组 。 组 号 为 0 表示 整个 表达 式 ， 
组 号 1 表示 被 第 一 对 括号 括 起 的 组 ， 依 此 类 推 。 因 此 ， 在 下 面 这 个 表达 式 ， 

A(B(C))D 
中 有 三 个 组 : 组 0 是 ABCD， 组 1 是 BC， 组 2 是 C。 

Matcher 对 象 提供 了 一 系列 方法 ， 用 以 获取 与 组 相关 的 信息 : public int groupCount(0 返 回 
该 匹配 器 的 模式 中 的 分 组 数目 ， 第 0 组 不 包括 在 内 。public String group) 返回 前 一 次 匹配 操作 
(例如 find0) 的 第 0 组 (整个 匹配 )。public String group(int i) 返回 在 前 一 次 匹配 操作 期 间 指 定 
的 组 号 ， 如 果 匹 配 成 功 ， 但 是 指定 的 组 没有 匹配 输入 字符 串 的 任何 部 分 ， 则 将 会 返回 null。 
public int start(int group) 返 回 在 前 一 次 匹配 操作 中 寻找 到 的 组 的 起 始 索引 。public int end(int 
group) 返 回 在 前 一 次 匹配 操作 中 寻找 到 的 组 的 最 后 一 个 字符 索引 加 一 的 值 。 

下 面 是 正则 表达 式 组 的 例子 : 


//: strings/Groups. java 
import java.util.regex.*: 
import static net.mindview.util.Print.*; 
public class Groups { 
static public final String POEM = 
"Twas brillig, and the slithy toves\n" + 
"Did gyre and gimble in the wabe.\n" + 
"ALL mimsy were the borogoves,\n" + 
"And the mome raths outgrabe.\n\n" + 
"Beware the Jabberwock. my son,\n* + 
"The jaws that bite, the claws that catch.\n” + 
"Beware the Jubjub bird, and shun\n" + 
"The frumious Bandersnatch."; 
public static void main(String[] args) { 
Matcher m = 
Pattern. compi te(" (2m) (\\S+)\\s+((\\S+)\\s+(\\S+))$") 
„matcher (POEM) ; 
while(m.find()) { 
for(int j = @; j <= m.groupCount(); j++) 
printnb("[" + m.group(j) + "J"); 
print(); 
} 


} 
) /* Output: 
[the slithy toves] [the] [slithy toves] [slithy] [toves] 
[in the wabe.] [in] (the wabe. ] [the] [wabe.] 
[were the borogoves, ] [were] [the 
borogoves ,] [the] [borogoves , ] 
[mome raths outgrabe. ] (mome] [raths 
outgrabe. ] [raths] [outgrabe.] 
(Jabberwock, my son,] [Jabberwock,] [my son, ] [my] [son,] 
[claws that catch.} [claws] {that catch.} [that] [catch.] 
[bird, and shun] [bird,] [and shun] (and) [shun] 
[The frumious Bandersnatch.} [The] [frumious 
Bandersnatch.] [frumious] [Bandersnatch.] 
“Ii~ 


这 首 诗 来 自 于 Lewis Carrollff) (Through the Looking Glass》 中 的 Jabberwocky。 可 以 看 到 这 
个 正则 表达 式 模式 有 许多 圆 括号 分 组 ， 由 任意 数目 的 非 空格 字符 (\S+) 及 随后 的 任意 数目 的 空 
格 字符 s+) 所 组 成 。 目 的 是 捕获 每 行 的 最 后 3 个 词 ， 每 行 最 后 以 $ 结 束 。 不 过 ， 在 正常 情况 下 
是 将 $ 与 整个 输入 序列 的 末端 相 匹配 。 所 以 我 们 一 定 要 显 式 地 告知 正则 表达 式 注意 输入 序列 中 的 
换行 符 。 这 可 以 由 序列 开头 的 模式 标记 (?m) 来 完成 (模式 标记 马上 就 会 介绍 )。 
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练习 12: (5) 修改 Goupsjava 类 ， 找 出 所 有 不 以 大 写字 母 开头 的 词 ， 不 重复 地 计算 其 个 数 。 

start() 与 end() 

在 匹配 操作 成 功 之 后 ，start0 返 回 先前 匹配 的 起 始 位 置 的 索引 ， 而 end0 返 回 所 匹配 的 最 后 
字符 的 索引 加 一 的 值 。 匹 配 操作 失败 之 后 (或 先 于 一 个 正在 进行 的 匹配 操作 去 尝试 ) 调用 start0 
或 end() 将 会 产生 IllegalStateException。 下 面 的 示例 还 同时 展示 了 matches0 和 lookingAt0 的 
MAS: 

41: strings/StartEnd. java 


import java.util.regex.*: 
import static net.mindview.util.Print.*; 


public class StartEnd { 
public static String input = 
"As long as there is injustice, whenever a\n" + 
"Targathian baby cries out, wherever a distress\n" + 
"signal sounds among the stars ... We'll be there.\n" + 
"This fine ship, and this fine crew ...\n" + 
"Never give up! Never surrender!" ; 
private static class Display { 
private boolean regexPrinted = false; 
private String regex: 
Display(String regex) { this.regex = regex; } 
void display(String message) { 
if (iregexPrinted) { 
print (regex) ; 
regexPrinted = true; 


} 
print (message) ; 


} 
static void examine(String s, String regex) { 
Display d = new Display (regex); 
Pattern p = Pattern. compile(regex) ; 
Matcher m = p.matcher(s); 
whi le(m.find()) 
d.display("find() '" + m.group() + 
"' start = "+ m.start() + ”end = " + m.end()); 
if (m.lookingAt()) // No reset() necessary 
d.display("lookingAt() start = * 
+ mstart() +" end = " + mend()); 
if(m.matches()) // No reset() necessary 
d.display("matches() start = " 
+m.start() + " end = ”+ mend()); 





} 
public static void matn(String[] args) { 
for(String in : input.split("\n")) { 
print("input : ”+ in); 
for (String regex : new String{]{"\\wtere\\w*", 
"\\weever", "T\\we", "Never.*?!"}) 
examine(in, regex); 





} 


} 
} /* Output: 
input : As long as there is injustice, whenever a 
\wrere\w* 
find() ‘there’ start = 11 end = 16 
\wrever 
find() ‘whenever’ start = 31 end = 39 
input : Targathian baby cries out, wherever a distress 
\wrere\w* 
find() ‘wherever’ start = 27 end = 35 


© 引用 自 电影 Galaxy Quest 中 Taggart 司 令 的 一 篇 演讲 。 
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\wtever 


find() ‘wherever’ start = 27 end = 35 


T\wt 


find() ‘Targathian’ start = @ end = 10 
lookingAt() start = 9 end = 16 
input : signal sounds among the stars ... We'll be there. 


\wtere\w* 


find() ‘there’ start = 43 end = 48 
input : This fine ship, and this fine crew ... 


T\wt 


find() 'This' start = O end = 4 
lookingAt() start = @ end = 4 
input : Never give up! Never surrender! 


\w*ever 


find() 'Never' start = @ end = 5 
find() ‘Never’ start = 15 end = 20 
lookingAt() start = @ end = 5 


Never. *?! 


find() ‘Never give up!' start = @ end = 14 


find() ‘Never surrender! ' 


start = 15 end = 31 


lookingAt() start = @ end = 14 


matches() start = 0 end = 31 


Whim 


注意 ，find0 可 以 在 输入 的 任意 位 置 定 位 正则 表达 式 ， 而 lookingAtO 和 matches0 只 有 在 正则 
表达 式 与 输入 的 最 开始 处 就 开始 匹配 时 才 会 成 功 。matches0 只 有 在 整个 输入 都 匹配 正则 表达 式 
时 才 会 成 功 ， 而 lookingAt0 只 要 输入 的 第 一 部 分 匹配 就 会 成 功 。 

练习 13; (2) 修改 StartEnd.java， 让 它 使 用 Groups.POEM 为 输入 ， 必 要 时 修改 正则 表达 式 ， 
使 find0、lookingAt0 和 matches0 都 有 机 会 匹配 成 功 。 


Pattern 标 记 


Pattern 类 的 compile0 方 法 还 有 另 一 个 版 本 ， 它 接受 一 个 标记 参数 ， 以 调整 匹配 的 行为 
Pattern Pattern.compile(String regex, int flag) 


其 中 的 fag 来 自 以 下 Pattern 类 中 的 常量 


编译 标记 


效 果 





Pattern.CANON_EQ 


Pattern.CASE_INSENSITIVE(?i) 


Pattern.COMMENTS(?x) 


Pattern.DOTALL(?s) 


Pattern. MULTILINE(?m) 


PFS LIL CNREE RHC, BRU EAA DARN. GA 
如 ， 如 果 我 们 指定 这 个 标记 ， 表 达 式 aw030A 就 会 匹配 字符 申 ?。 在 默认 的 情况 
下 ,匹配 不 考虑 规范 的 等 价 性 。 

默认 情况 下 ， 大 小 写 不 敏感 的 匹配 假定 只 有 US-ASCII 字 符 集中 的 字符 才能 
进行 。 这 个 标记 允许 模式 匹配 不 必 考 虑 大 小 写 (大 写 或 小 写 ) 。 通 过 指定 
UNICODE_CASE 标记 及 结合 此 标记 ， 基 于 Unicode 的 大 小 写 不 敏感 的 匹配 就 
可 以 开启 了 

在 这 种 模式 下 ， 空 格 符 将 被 忽略 掉 ， 并 且 以 # 下 始 直到 行 未 的 注释 也 会 被 忽 
略 挤 。 通 过 嵌入 的 标记 表达 式 也 可 以 开启 Unix 的 行 模式 

在 dotall 模 式 中 ， 表 达 式 “.” 匹 配 所 有 字符 ， 包 括 行 终结 符 。 默 认 情况 下 ， 
“-” 表 达 式 不 匹配 行 终结 符 

在 多 行 模式 下 ， 表 达 式 ^ 和 $ 分 别 匹配 一 行 的 开始 和 结束 。^ 还 匹配 输入 字符 
串 的 开始 ， 而 $ 还 匹配 输入 字符 串 的 结尾 。 默 认 情 况 下 ， 这 些 表达 式 仅 匹配 输 
人 的 完整 字符 串 的 开始 和 结束 








O 我 完全 不 理解 设计 师 怎 么 会 想到 这 么 个 方法 名 。 不 过 可 以 相信 ， 取 出 如 此 不 直观 的 名 字 的 家 伙 还 在 Sun 工 作 。 
而 且 ， 不 复查 代码 设计 的 政策 显然 还 存在 于 Sun 中 。 请 原谅 我 的 讽刺 ， 但 是 多 年 来 ， 同 样 的 事情 一 再 发 生 ， 这 
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( 续 ) 
编译 标记 效 R 
Pattern.UNICODE_CASE(?u) 当 指定 这 个 标记 ， 并 且 开启 CASE_INSENSITIVE 时 ， 大 小 写 不 敏感 的 匹配 
将 按照 与 Unicode 标 准 相 一 致 的 方式 进行 。 默 认 情 况 下 ， 交 小写 不 敏感 的 匹配 
假定 只 能 在 US-ASCII 字符 集中 的 字符 才能 进行 
Pattern.UNIX_LINES(?d) 在 这 种 模式 下 ， 在 .、^ 和 $ 行 为 中 ， 只 识别 行 终结 符 n 
在 这 些 标记 中 ，Pattern.CASE_INSENSITIVE、Pattern.MULTILINE 以 及 Pattern. 
COMMENTS (对 声明 或 文档 有 用 ) 特别 有 用 。 请 注意 ， 你 可 以 直接 在 正则 表达 式 中 使 用 其 中 
的 大 多 数 标记 ， 只 需要 将 上 表 中 括号 括 起 的 字符 插入 到 正则 表达 式 中 ， 你 希望 它 起 作用 的 位 置 
即 可 。 
你 还 可 以 通过 “或 ”(1) 操作 符 组 合 多 个 标记 的 功能 : 


//: strings/ReFlags. java 
import java.util.regex.*; 








public class ReFlags { 
public static void main(String{] args) { 
Pattern p = Pattern.compile("^java", 
Pattern.CASE_INSENSITIVE | Pattern. MULTILINE); 
Matcher m = p.matcher( 
“java has regex\nJava has regex\n" + 
"JAVA has pretty good regular expressions\n" + 
"Regular expressions are in Java"); 
while(m.find()) 
System.out.printIn(m.group()); 


} 
} /* Output: 
java 
Java 
JAVA 
Whim 


在 这 个 例子 中 ,我们 创建 了 一 个 模式 ， 它 将 匹配 所 有 以 “java”"、“Java” 和 “JAVA” 等 开头 的 行 ， 
并 且 是 在 设置 了 多 行 标记 的 状态 下 ， 对 每 一 个 行 〈 从 字符 序列 的 第 一 个 字符 开始 ， 至 每 一 个 行 
终结 符 ) 都 进行 匹配 。 注 意 ，group( 方 法 只 返回 已 匹配 的 部 分 。 


13.6.5 split) 
split0 方 法 将 输入 字符 串 断 开 成 字符 串 对 象 数组 ， 断 开 边 界 由 下 列 正则 表达 式 确定 : 


Stringi] split(CharSequence input) 
String[] split(CharSequence input, int limit) 


这 是 一 个 快速 而 方便 的 方法 ， 可 以 按照 通用 边界 断 开 输入 文本 : 


//: strings/SplitDemo. java 
import java.util.regex.*; 

import java.util.*; 

import static net.mindview.util.Print.*; 


public class SplitDemo { 
public static void main(String{} args) { 
String input = 
"This! !unusual use! !of exclamation! !points": 
print (Arrays. toString 
Pattern. compile("!!").split(input))); 
// Only do the first three: 
print (Arrays. toString( 
Pattern.compile("!!").split(input. 3))); 
} 








539) 
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} 7* Output: 

[This, unusual use, of exclamation, points] 
(This, unusual use, of exclamation! !points} 
Wh 


第 二 种 形式 的 split0 方 法 可 以 限制 将 输入 分 割 成 字符 串 的 数量 。 
练习 14: (1) 用 String.split0 重 写 SplitDemo。 


13.6.6 蔡 换 操作 

正则 表达 式 特别 便于 替换 文本 ， 它 提供 了 许多 方法 : replaceFirst(String replacement) 以 参 
数字 符 串 replacement 替 换 掉 第 一 个 匹配 成 功 的 部 分 。replaceAll(String replacement) 以 参数 字符 
串 replacement 替 换 所 有 匹配 成 功 的 部 分 。appendReplacement(StringBuffer sbuf, String 
replacement) 执行 渐进 式 的 替换 ， 而 不 是 像 replaceFirst0 和 replaceAlIO 那 样 只 替换 第 一 个 匹配 或 
全 部 匹配 。 这 是 一 个 非常 重要 的 方法 。 它 允许 你 调用 其 他 方法 来 生成 或 处 理 replacement 
(replaceFirst0 和 replaceAlI0 则 只 能 使 用 一 个 固定 的 字符 串 )， 使 你 能 够 以 编程 的 方式 将 目标 分 
割 成 组 ， 从 而 具备 更 强大 的 替换 功能 。appendTail(StringBuffer sbuf)， 在 执行 了 一 次 或 多 次 
appendReplacement0 之 后 ， 调 用 此 方法 可 以 将 输入 字符 串 余下 的 部 分 复制 到 sbuf 中 。 

下 面 的 程序 演示 了 如 何 使 用 这 些 替 换 方 法 。 开 头 部 分 注释 掉 的 文本 ， 就 是 正则 表达 式 要 处 
理 的 输入 字符 串 。 


//: strings/TheReplacements. java 

import java.util.regex. 

import net.mindview.util.*; 

import static net.mindview.util.Print.*; 


/*! Here's a block of text to use as input to 
the regular expression matcher. Note that we'll 
first extract the block of text by looking for 
the special delimiters, then process the 
extracted block. !*/ 


public class TheReplacements { 
public static void main(String{] args) throws Exception { 
String s = TextFile.read("TheReplacements. java"); 
// Match the specially commented block of text above: 
Matcher mInput = 
Pattern.compile("/\\*!(.*)!\\*/", Pattern. DOTALL) 
„matcher (s); 
if(mInput. find()) 
s = mInput.group(1); // Captured by parentheses 
// Replace two or more spaces with a single space: 
s = s.replaceAll(” {2,)", " "); 
// Replace one or more spaces at the beginning of each 
// Vine with no spaces. Must enable MULTILINE mode: 
s = s.replaceALl("(?m)* +", ""); 
print(s); 
s = s.replaceFirst("{aeiou)*, *(VOWEL1)"): 
StringBuffer sbuf = new StringBuffer(); 
Pattern p = Pattern.compite(*[aetou]"); 
Watcher m = p.matcher(s); 
// Process the find information as you 
// perform the replacements: 
wht le(m.find()) 
m.appendReplacement (sbuf, m.group().toUpperCase()); 
// Put in the remainder of the text: 
m. appendTail(sbuf) ; 
print(sbuf); 
} 
} /* Output: 
Here's a block of text to use as input to 
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the regular expression matcher. Note that we'll 
first extract the block of text by looking for 
the special delimiters, then process the 
extracted block. 

H(VOWEL1)rE's A blOck Of tExt tO USE As InpUt to 
thE rEgULAr ExprEssIOn mAtchEr. NOtE thAt we'll 
flrst Extract thE blOck Of tExt by 100kIng fOr 
thE spEcIAL dELImItErs，thEn prOcEss the 
ExtrActEd blOck. 

Wh 


此 处 使 用 TextFile 类 打开 并 读 和 文件， 该 类 在 netmindview.util 工 具 包 中 (在 第 18 章 中 对 其 
代码 有 详细 介绍 )。static read( 方 法 读 人 整个 文件 ， 将 其 内 容 作 为 String 对 象 返回 。mInput 用 以 
匹配 在 /*! 和 !*/ 之 间 的 所 有 文字 (注意 分 组 的 括号 )。 接 下 来 , 将 存在 两 个 或 两 个 以 上 空格 的 地 方 ， 
缩减 为 一 个 空格 ， 并 且 删 除 每 行 开 头 部 分 的 所 有 空格 (为 了 使 每 一 行 都 达到 这 个 效果 ， 而 不 仅 
仅 只 是 删除 文本 开头 部 分 的 空格 ， 这 里 特意 打开 了 多 行 状态 ) 。 这 两 个 赫 换 操作 所 使 用 的 
replaceAllO 是 String 对 象 自 带 的 方法 ， 在 这 里 ， 使 用 此 方法 更 方便 。 注 意 ， 因 为 这 两 个 替换 操作 
都 只 使 用 了 一 次 replaceAlHO， 所 以 ， 与 其 编译 为 Pattern， 不 如 直接 使 用 String 的 replaceAll( 方 
法 ， 而 且 开销 也 更 小 些 。 

replaceFirstO 只 对 找到 的 第 一 个 匹配 进行 替换 。 此 外 ，replaceFirst0 和 repalceAl( 方 法 用 来 
替换 的 只 是 普通 的 字符 种 ， 所 以 ， 如 果 想 对 这 些 替 换 字符 串 执行 某 些 特殊 处 理 ， 这 两 个 方法 是 
无 法 胜任 的 。 如 果 你 想 要 那么 做 ， 就 应 该 使 用 appendReplacement0 方 法 。 该 方法 允许 你 在 执行 
赫 换 的 过 程 中 ， 操 作用 来 替换 的 字符 串 。 在 这 个 例子 中 ， 先 构造 了 sbuf 用 来 保存 最 终结 果 ， 然 
后 用 group0 选 择 一 个 组 ， 并 对 其 进行 处 理 ， 将 正则 表达 式 找 到 的 元 音字 母 转换 成 大 写字 母 。 一 
般 情 况 下 ， 你 应 该 遍历 执行 所 有 的 替换 操作 ， 然 后 再 调用 appendTail0 方 法 ， 但 是 ， 如 果 你 想 模 
ireplaceFirst() (或 替换 n 次 ) 的 行为 ， 那 就 只 需 执行 一 次 替换 ， 然 后 调用 appendTail0 方 法 ， 
将 剩余 未 处 理 的 部 分 存 人 sbuf 即 可 。 

同时 ，appendRepelacement( 方 法 还 允许 你 通过 $g 直 接 找到 匹配 的 某 个 组 ， 这 里 的 g 就 是 组 
号 。 然 而 ， 它 只 能 应 付 一 些 简单 的 处 理 ， 无 法 实现 类 似 前 面 这 个 例子 中 的 功能 。 

13.6.7 reset() 

通过 reset0 方 法 ， 可 以 将 现 有 的 Mateher 对 象 应 用 于 一 个 新 的 字符 序列 : 

//: strings/Resetting.java 

import java.util.regex.*; 


public class Resetting { t 
public static void main(String[] args) throws Exception { 
Matcher m = Pattern.compile("[frb] [aiu] [gx] ") 
„matcher ("fix the rug with bags"); 
wht Le(m. find()) 
System.out.print(m.group() +" "); 
system.out.printin(): 
m.reset("fix the rig with rags"); 
whi le(m. findo) 
System.out.print(m.group() + * "); 


} 
} /* Output: 
fix rug bag 
tix rig rag 
#7:~ 


使 用 不 带 参 数 的 reset0 方 法 ， 可 以 将 Matcher 对 象 重新 设置 到 当前 字符 序列 的 起 始 位 置 。 
13.6.8 正则 表达 式 与 Java VO 
到 目前 为 止 ， 我们 看 到 的 例子 都 是 将 正则 表达 式 应 用 于 静态 的 字符 串 。 下 面 的 例子 将 向 你 
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演示 ， 如 何 应 用 正则 表达 式 在 一 个 文件 中 进行 搜索 匹配 操作 。JGrep.java 的 灵感 源 自 于 Unix 上 
的 grep。 它 有 两 个 参数 : 文件 名 以 及 要 匹配 的 正则 表达 式 。 输 出 的 是 有 匹配 的 部 分 以 及 匹配 部 
分 在 行 中 的 位 置 。 


//: strings/JGrep. java 
11 A very simple version of the “grep program. 
// {Args: JGrep. java "\\b(Ssct]\\we"} 

import java.util.regex.*; 

import net.mindview.util.*; 


public class JGrep { 
public static void main(String[] args) throws Exception { 
if(args. length < 2) { 
System.out.printin("Usage: java JGrep file regex"); 
System. exit(@); 
} 
Pattern p = Pattern.compile(args(1]); 
71 Iterate through the lines of the input file: 
int index = 0; 
Matcher m = p.matcher(""); 
for(String line : new TextFite(args{@})) { 
m.reset (line); 
while (m. find()) 
System.out.printin(index++ +": " + 
m.group() + ": " + mstart()); 
} 


} 

} /* Output: (Sample) 
@: strings: 4 
1: simple: 10 
2: the: 28 

3: Ssct: 26 

4: class: 7 

5: static: 9 
6: String: 26 
7: throws: 41 
8: System: 6 
9: System: 6 
10: compile: 24 
11: through: 15 
12: the: 23 

13: the: 36 
14: String: 8 
15: System: 8 
16: start: 31 


通过 net.mindview.util.TextFile 对 象 将 文件 打开 (在 第 18 章 中 有 详细 的 介绍 )， 读 入 所 有 的 行 
后 ， 并 存储 在 一 个 ArrayList 中 。 因 此 ， 可 以 用 循环 来 迭代 遍历 TextFile 对 象 中 的 所 有 行 。 虽 然 也 
可 以 在 for 循 环 内 部 创建 新 的 Matcher 对 象 ， 但 是 ， 在 循环 外 创建 一 个 空 的 Matcher 对 象 ， 然 后 用 
reset() 方 法 每 次 为 Matcher 加 载 一 行 输入 ， 这 种 处 理会 有 一 定 的 性 能 优化 。 最 后 用 find0 搜 索 结 
果 。 这 里 读 人 的 测试 参数 是 JGrep.java 文 件 ， 然 后 搜索 以 [Ssct] 开 头 的 单词 。 

如 果 想 要 更 深入 的 学 习 正 则 表达 式 ， 你 可 以 阅读 Jeffrey E. F Friedl 的 《精通 正则 表达 式 (第 
?2 版)》。 网 络 上 也 有 很 多 正则 表达 式 的 介绍 ， 你 还 可 以 从 Ped 和 Python 等 其 他 语言 的 文档 中 找到 
有 用 的 信息 。 

练习 15; (5) 修改 JGrep.java 类 ， 令 其 能 够 接受 模式 标志 参数 (例如 Pattern.CASE_ 
INSENSITIVE, Pattern.MULTILINE). 

练习 16: (5) 修改 JGrep.java 类 ， 令 其 能 够 接受 一 个 目录 或 文件 为 参数 (如 果 传人 的 是 目录 ， 
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就 搜索 目录 中 的 所 有 文件 )。 提 示 : 可 以 用 下 面 的 方法 获得 所 有 文件 的 名 字 列 表 : 

Filel] files = new File(".").listFiles(); 

练习 17: (8) 编写 一 个 程序 ， 读 取 一 个 Java 源 代码 文件 (可 以 通过 控制 台 参 数 提供 文件 名 ) ， 
打印 出 所 有 注释 。 

练习 18: (8) 编写 一 个 程序 ， 读 取 一 个 Java 源 代码 文件 (可 以 通过 控制 台 参 数 提供 文件 名 ) ， 
打印 出 代码 中 所 有 的 普通 字符 串 。 

练习 19，(8) 在 前 两 个 练习 的 基础 上 ， 编 写 一 个 程序 ， 输 出 Java 源 代码 中 用 到 的 所 有 类 的 
名 字 。 
13.7 扫描 输入 


到 目前 为 止 ， 从 文件 或 标准 输入 读 取 数据 还 是 一 件 相当 痛苦 的 事情 。 一 般 的 解决 之 道 就 是 
读 入 一 行文 本 ， 对 其 进行 分 词 ， 然 后 使 用 Integer、Double 等 类 的 各 种 解析 方法 来 解析 数据 : 


//: strings/SimpleRead. java 
import java.io.*; 
public class SimpleRead { 
public static BufferedReader input = new BufferedReader ( 
new StringReader ("Sir Robin of Camelot\n22 1.61803")); 
public static void main(String[] args) { 
try { 
System.out.printin("What is your name?"); 
String name = input.readLine(): 
System. out.printin(name) ; 
System.out.printin( 

“How old are you? What is your favorite double?"); 
System.out.printin("(input: <age> <double>)"); 
String numbers = input.readLine(); 

System. out.printin(numbers) ; 

String[] numArray = numbers.split(" "); 

int age = Integer.parseInt (numArray[0]) ; 

double favorite = Double.parseDouble(numArray[1]); 
System. out.format("Hi %s.\n", name); 
System.out.format("In 5 years you will be %d.\n", 

age + 5); 

System.out.format("My favorite double is %f.", 

favorite / 2); 
catch(IOException e) { 

System.err.printin("I/0 exception"); 
} 


} 
} /* Output: 
What is your name? 
Sir Robin of Camelot 
How old are you? What is your favorite double? 
(input: <age> <double>) 
22 1.61803 
Hi Sir Robin of Camelot. 
In 5 years you will be 27. 
My favorite double is 0.809015. 
Whim 


input 元 素 使 用 的 类 来 自 java.io， 在 第 18 章 中 我 们 会 正式 介绍 这 个 包 中 的 内 容 。StringReader 
将 String 转 化 为 可 读 的 流 对 象 ， 然 后 用 这 个 对 象 来 构造 BufferReader 对 象 ， 因 为 我 们 要 使 用 
BufferReader 的 readLine0 方 法 。 最 终 ， 我 们 可 以 使 用 input 对 象 一 次 读 取 一 行文 本 ， 就 像 是 从 控 
制 台 读 人 标准 输入 一 样 。 

readLine0 方 法 将 一 行 输入 转 为 String 对 象 。 如 果 每 一 行 数据 正好 对 应 一 个 输入 值 ， 那 这 个 
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方法 也 还 可 行 。 但 是 ， 如 果 两 个 输入 值 在 同一 行 中 ， 事 情 就 不 好 办 了 ， 我 们 必须 分 解 这 个 行 ， 
才能 分 别 翻译 所 需 的 输入 值 。 在 这 个 例子 中 , 分 解 的 操作 发 生 在 创建 numArray 时 ， 不 过 要 注意 ， 
split0 方 法 是 J2SE1.4 中 的 方法 ， 所 以 在 这 之 前 ， 你 还 必须 做 些 别 的 准备 。 

终于 ，Java SE5 新 增 了 Scanner 类 ， 它 可 以 大 大 减轻 扫描 输入 的 工作 负担 ， 


//: strings/BetterRead. java 
import java.util.*; 


public class BetterRead { 
public static void main(String{] args) { 
Scanner stdin = new Scanner (SimpleRead. input): 
System.out.printin("What is your name?"): 
String name = stdin.nextLine(); 
System. out.print1n(name) ; 
System.out .prtnttn( 
"How old are you? What is your favorite double?"); 
System.out.printin("(input: <age> <double>)"); 
int age = stdin.nextInt(); 
double favorite = stdin.nextDouble(); 
System.out.printtn(age) ; 
System. out. printin(favorite) ; 
System.out. format ("Hi %s.\n", name): 
System.out.format("In 5 years you will be %d.\n", 
age + 5); 
System.out.format("My favorite double is %f.", 
favorite / 2); 


} 
} /* Output: 
What. is your name? 
Sir Robin of Camelot 
How old are you? What is your favorite double? 
(input: <age> <double>) 
22 


1.61803 

Hi Sir Robin of Camelot. 

In 5 years you will be 27. 

My favorite double is 6.899015. 
Wh 


Scanner 的 构造 器 可 以 接受 任何 类 型 的 输入 对 象 ， 包 括 File 对 象 同样， 我 们 将 在 第 18 章 中 
详细 介绍 File 类 )、InputStream、String 或 者 像 此 例 中 的 Readable 对 象 。Readable 是 Java SE5 中 
新 加 入 的 一 个 接口 ， 表 示 “ 具 有 read0 方 法 的 某 种 东西 "。 前 一 个 例子 中 的 BufferedReader 也 归 
于 这 一 类 。 有 了 Scanner， 所 有 的 输入 、 分 词 以 及 翻译 的 操作 都 隐藏 在 不 同类 型 的 next 方 法 中 。 
普通 的 next0 方 法 返回 下 一 个 String。 所 有 的 基本 类 型 ( 除 char 之 外 ) 都 有 对 应 的 next 方 法 ， 包 
括 BigDecimal 和 BigInteger。 所 有 的 next 方 法 ， 只 有 在 找到 一 个 完整 的 分 词 之 后 才 会 返回 。 
Scanner 还 有 相应 的 hasNext 方 法 ， 用 以 判断 下 一 个 输入 分 词 是 否 所 需 的 类 型 。 

在 前 面 的 两 个 例子 中 ， 一 个 有 趣 的 区 别 是 ，BetterRead,java 没 有 针对 IOException 添 加 try 区 
块 。 因 为 ，Scanner 有 一 个 假设 ， 在 输入 结束 时 会 抛 出 IOException， 所 以 Scanner 会 把 
IOException 吞 掉 。 不 过 ， 通 过 ioException0 方 法 ， 你 可 以 找到 最 近 发 生 的 异常 ， 因 此 ， 你 可 以 
在 必要 时 检查 它 。 

练习 20，(2) 编写 一 个 包含 int、long、float、double 和 String 属 性 的 类 。 为 它 编写 一 个 构造 
器 ， 接 收 一 个 String 参 数 。 然 后 扫描 该 字符 串 ， 为 各 个 属性 赋值 。 再 添加 一 个 toString() 方 法 ， 
用 来 演示 你 的 类 是 否 工作 正确 。 

13.7.1 Scanner 定 界 符 
在 默认 的 情况 下 ，Scanner 根 据 空白 字符 对 输入 进行 分 词 ， 但 是 你 可 以 用 正则 表达 式 指定 自 
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己 所 需 的 定 界 符 : 


//: strings/ScannerDelimiter.java 
import java.util.*; 


public class ScannerDelimiter { 
public static void main(String[] args) { 
Scanner scanner = new Scanner("12, 42, 78, 99, 42"); 
scanner .useDelimiter("\\s*,\\s*"); 
while (scanner.hasNextInt()) 
System.out.printIn(scanner.nextInt()); 


} 
}_/* Output: 
12 


这 个 例子 使 用 逗号 (包括 逗号 前 后 任意 的 空白 字符 ) 作为 定 界 符 ， 同 样 的 技术 也 可 以 用 来 
读 取 逗号 分 隔 的 文件 。 我 们 可 以 用 useDelimiter0 来 设置 定 界 符 , 同时 ,还 有 一 个 delimiter0 方 法 ， 
用 来 返回 当前 正在 作为 定 界 符 使 用 的 Pattern 对 象 。 


13.7.2 用 正则 表达 式 扫描 
除了 能 够 扫描 基本 类 型 之 外 ， 你 还 可 以 使 用 自 定义 的 正则 表达 式 进行 扫描 ， 这 在 扫描 复杂 
数据 的 时 候 非常 有 用 。 下 面 的 例子 将 扫描 一 个 防火 墙 日 志文 件 中 记录 的 威胁 数据 : 


//: strings/ThreatAnalyzer ,java 
import java.util.regex.*; 
import java.util.*; 


public class ThreatAnalyzer { 
static String threatData = 
"58.27.82.161@02/10/2605\n" + 
4 .40@02/11/2805\n" + 
1@02/11/2005\n" + 
161@02/12/2005\n" + 
"58.27.82. 161@02/12/2005\n" + 
"(Next log section with different data format]"; 
public static void main(String{] args) { 
Scanner scanner = new Scanner (threatData) ; 
String pattern = "(\\d+[.]\\d+[.]\\d+[.]\\d+)e" + 
" (\Nd{2}/\Nd{2)/\\0 4} ) "5 
while(scanner.hasNext(pattern)) { 
scanner .next (pattern); 
MatchResult match = scanner.match(); 
String ip = match. group(1) ; 
String date = match.group(2); 
System.out.format("Threat on %s from %s\n", date, ip); 
$ 





} 

} /* Output: 

Threat on 02/19/2005 from 58.27.82.161 
Threat on 02/11/2005 from 204.45.234.40 
Threat on 02/11/2005 from 58.27.82.161 
Threat on 02/12/2005 from 58.27.82.161 
Threat on 92/12/2005 from 58.27.82.161 
Ii~ 


当 next0 方 法 配合 指定 的 正则 表达 式 使 用 时 ， 将 找到 下 一 个 匹配 该 模式 的 输入 部 分 ， 调 用 
match0 方 法 就 可 以 获得 匹配 的 结果 。 如 上 所 示 ， 它 的 工作 方式 与 之 前 看 到 正则 表达 式 匹配 相似 。 
在 配合 正则 表达 式 使 用 扫描 时 ， 有 一 点 需要 注意 : 它 仅 仅 针对 下 一 个 输入 分 词 进行 匹配 ， 如 果 
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你 的 正则 表达 式 中 含有 定 界 符 ， 那 永远 都 不 可 能 匹配 成 功 。 
13.8 StringTokenizer 


在 Java 引 入 正则 表达 式 (J2SE1.4) 和 Scanner 类 (Java SES) 之 前 ， 分 割 字符 串 的 唯一 方法 
是 使 用 StringTokenizer 来 分 词 。 不 过 ， 现 在 有 了 正则 表达 式 和 Scanner， 我 们 可 以 使 用 更 加 简单 、 
更 加 简洁 的 方式 来 完成 同样 的 工作 了 。 下 面 的 例子 中 ， 我 们 将 StringTokenizer 与 另 两 种 技术 做 
了 一 个 比较 ， 


//: strings/ReplacingStringTokenizer . java 
import java.util.*; 


public class ReplacingStringTokenizer { 
public static void main(String{] args) { 
String input = “But I'm not dead yet! I feel hap 
StringTokenizer stoke = new StringTokenizer (input); 
while(stoke.hasMoreElements()) 
System.out.print(stoke.nextToken() + " "); 
System.out.printin(); 
System. out..printin(Arrays.toString(input.split(" "))); 
Scanner scanner = new Scanner (input); 
while (scanner. hasNext()) 
System.out.print(scanner.next() + " "); 








£ 
} /* Output: 
But I'm not dead yet! I feel happy! 
[But, I'm, not, dead, yet!, I, feel, happy!) 
But I'm not dead yet! I feel happy! 
Whim 


使 用 正则 表达 式 或 Scanner 对 象 ， 我 们 能 够 以 更 加 复杂 的 模式 来 分 割 一 个 字符 种 ， 而 这 对 于 
StringTokenizer 来 说 就 很 困难 了 。 基 本 上 ， 我 们 可 以 放心 的 说 ，StringTokenizer 已 经 可 以 废弃 
不 用 了 。 


13.9 总 结 


过 去 ，Java 对 字符 串 操作 的 支持 相当 不 完善 。 不 过 随 着 近 几 个 版 本 的 升级 ， 我 们 可 以 看 到 ， 
Java 已 经 从 其 他 语言 中 吸取 了 许多 成 熟 的 经 验 。 到 目前 为 止 ， 它 对 字符 串 操作 的 支持 已 经 很 完 
善 了 。 不 过 ， 有 时 你 还 要 在 细节 上 注意 效率 的 问题 ， 例 如 恰当 地 使 用 StringBuilder 等 。 

所 选 习题 的 答案 都 可 以 在 名 为 The Thinking in Java Annotated Solution Guide 的 电子 文档 中 找 


到 ， 读 者 可 以 从 www.MindView.net 处 购买 此 文档 。 





第 14 章 类 型 信息 


运行 时 类 型 信息 使 得 你 可 以 在 程序 运行 时 发 现 和 使 用 类 型 信息 。 

它 使 你 从 只 能 在 编译 期 执行 面向 类 型 的 操作 的 禁 铀 中 解脱 了 出 来 ， 并 且 可 以 使 用 某 些 非常 
强大 的 程序 。 对 RTTI 的 需要 ， 揭 示 了 面向 对 象 设计 中 许多 有 趣 (并 且 复 杂 ) 的 问题 ， 同 时 也 提 
出 了 如 何 组 织 程序 的 问题 。 

本 章 将 讨论 Java 是 如 何 让 我 们 在 运行 时 识别 对 象 和 类 的 信息 的 。 主 要 有 两 种 方式 : 一 种 是 
“传统 的 ”RTTI， 它 假定 我 们 在 编译 时 已 经 知道 了 所 有 的 类 型 ， 另 一 种 是 “反射 ”机 制 ， 它 允许 
我 们 在 运行 时 发 现 和 使 用 类 的 信息 。 


14.1 为 什么 需要 RTTI 


下 面 看 一 下 我 们 已 经 很 熟悉 了 的 一 个 例子 ， 它 使 用 了 多 态 的 类 层次 结构 。 最 通用 的 类 型 
(W) 是 基 类 Shape， 而 派生 出 的 具体 类 有 Circle、 = 





























Square 和 Triangle ( 见 右 图 所 示 )。 are 
这 是 一 个 典型 的 类 层次 结构 图 ， 基 类 位 于 顶部 ， 派 
生 类 向 下 扩展 。 面 向 对 象 编程 中 基本 的 目的 是 ; 让 代码 ] 
只 操纵 对 基 类 (这 里 是 Shape) 的 引用 。 这 样 ， 如 果 要 
Circle Square | | Triangle 











添加 一 个 新 类 (比如 从 Shape 派 生 的 Rhomboid) 来 扩展 
程序 ， 就 不 会 影响 到 原来 的 代码 。 在 这 个 例子 的 Shape 接 口中 动态 绑 定 了 draw0 方 法 ， 目 的 就 是 
让 客户 端 程序 员 使 用 泛 化 的 Shape 引 用 来 调用 draw0。draw0 在 所 有 派生 类 里 都 会 被 覆盖 ， 并且 
由 于 它 是 被 动态 绑 定 的 ， 所 以 即使 是 通过 泛 化 的 Shape 引 用 来 调用 ， 也 能 产生 正确 行为 。 这 就 
是 多 态 。 

因此 ， 通 常会 创建 一 个 具体 对 象 (Circle，Square， 或 者 Triangle) ， 把 它 向 上 转型 成 Shape 
(忽略 对 象 的 具体 类 型 )， 并 在 后 面 的 程序 中 使 用 匿名 《译注 : 即 不 知道 具体 类 型 ) 的 Shape 引 用 。 

你 可 以 像 下 面 这 样 对 Shape 层 次 结构 编码 : 


//: typeinfo/Shapes. java 
import java.util. *; 





abstract class Shape { 
void draw() { System.out.printin(this + ".draw()"); } 
abstract public String toString(); 

} 


class Circte extends Shape { 
public String toString() { return "Circle": } 
} 


class Square extends Shape { 
public String toString() { return “Square”; } 


class Triangle extends Shape { 
public String toString() { return “Triangle”; } 
} 
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public class Shapes { 
public static void main(String[] args) { 
List<Shape> shapeList = Arrays.asList( 
new Circle(), new Square(), new Triangle() 


); 
for(Shape shape : shapeList) 
shape.draw(); 
} 
} /* Output: 
Cirele.draw() 
Square.draw() 
Triangle.draw() 
Whim 


基 类 中 包含 draw0 方 法 ， 它 通过 传递 this 参 数 给 System.outprintin0， 间 接地 使 用 toString0) 
打印 类 标识 符 (注意 ，toString0 被 声明 为 abstract， 以 此 强制 继承 者 覆 写 该 方法 ， 并 可 以 防止 对 
无 格式 的 Shape 的 实例 化 ) 。 如 果 某 个 对 象 出 现在 字符 串 表达 式 中 (涉及 “+” 和 字符 串 对 象 的 表 
达 式 )，toString0 方 法 就 会 被 自动 调用 ， 以 生成 表示 该 对 象 的 String。 每 个 派生 类 都 要 覆盖 (从 
Object 继 承 来 的 ) toString0 方 法 ， 这 样 draw0 在 不 同情 况 下 就 打印 出 不 同 的 消息 (多 态 ) 。 

在 这 个 例子 中 ， 当 把 Shape 对 象 放 人 List<Shape> 的 数组 时 会 向 上 转型 。 但 在 向 上 转型 为 
Shape 的 时 候 也 丢失 了 Shape 对 象 的 具体 类 型 。 对 于 数组 而 言 ， 它 们 只 是 Shape 类 的 对 象 。 

当 从 数组 中 取出 元 素 时 ， 这 种 容器 一 实际 上 它 将 所 有 的 事物 都 当 作 Object 持 有 一 一 会 自动 
将 结果 转型 回 Shape。 这 是 RTTI 最 基本 的 使 用 形式 ， 因 为 在 Java 中 ， 所 有 的 类 型 转换 都 是 在 运行 
时 进行 正确 性 检查 的 。 这 也 是 RTTI 名 字 的 含义 : 在 运行 时 ， 识 别 一 个 对 象 的 类 型 。 

在 这 个 例子 中 ，RTTI 类 型 转换 并 不 彻底 ，Object 被 转型 为 Shape， 而 不 是 转型 为 Cirele、 
Square 或 者 Triangle。 这 是 因为 目前 我 们 只 知道 这 个 List<Shape> 保 存 的 都 是 Shape。 在 编译 时 ， 
将 由 容器 和 Java 的 泛 型 系统 来 强制 确保 这 一 点 ， 而 在 运行 时 ， 由 类 型 转换 操作 来 确保 这 一 点 。 

接 下 来 就 是 多 态 机 制 的 事情 了 ，Shape 对 象 实际 执行 什么 样 的 代码 ， 是 由 引用 所 指向 的 具体 
对 象 Circle、Square 或 者 Triangle 而 决定 的 。 通 常 ， 也 正 是 这 样 要 求 的 ， 你 希望 大 部 分 代码 尽 可 
能 少 地 了 解 对 象 的 具体 类 型 ， 而 是 只 与 对 象 家 族 中 的 一 个 通用 表示 打交道 (在 这 个 例子 中 是 
Shape)。 这 样 代码 会 更 容易 写 ， 更 容易 读 ， 且 更 便于 维护 ， 设 计 也 更 容易 实现 、 理 解 和 改变 。 
所 以 “多 态 ”是 面向 对 象 编程 的 基本 目标 。 

但 是 ， 假 如 你 磁 到 了 一 个 特殊 的 编程 问题 一 如果 能 够 知道 某 个 泛 化 引用 的 确切 类 型 ， 就 
可 以 使 用 最 简单 的 方式 去 解决 它 ， 那 么 此 时 该 怎么 办 呢 ? 例 如， 假设 我 们 允许 用 户 将 某 一 具体 
类 型 的 几何 形状 全 都 变 成 某 种 特殊 的 颜色 ， 以 突出 显示 它们 。 通 过 这 种 方法 ， 用 户 就 能 找 出 屏 
幕 上 所 有 被 突出 显示 的 三 角形 。 或 者 ， 可 能 要 用 某 个 方法 来 旋转 列 出 的 所 有 图 形 ， 但 想 跳 过 圆 
形 ， 因 为 对 圆 形 进行 旋转 没有 意义 。 使 用 RTTI， 可 以 查询 某 个 Shape 引 用 所 指向 的 对 象 的 确切 
类 型 ， 然 后 选择 或 者 别 除 特例 。 


14.2 Class 对 象 


要 理解 RTTI 在 Java 中 的 工作 原理 ， 首 先 必须 知道 类 型 信息 在 运行 时 是 如 何 表示 的 。 这 项 工 
作 是 由 称 为 Class 对 象 的 特殊 对 象 完 成 的 ， 它 包含 了 与 类 有 关 的 信息 。 事 实 上 ，Class 对 象 就 是 用 
来 创建 类 的 所 有 的 “常规 ”对 象 的 。Java 使 用 Class 对 象 来 执行 其 RTTI， 即 使 你 正在 执行 的 是 类 
似 转型 这 样 的 操作 。Class 类 还 拥有 大 量 的 使 用 RTTI 的 其 他 方式 。 

类 是 程序 的 一 部 分 ， 每 个 类 都 有 一 个 Class 对 象 。 换 言 之 ， 每 当 编 写 并 且 编 译 了 一 个 新 类 ， 
就 会 产生 一 个 Class 对 象 ( 更 恰当 地 说 ， 是 被 保存 在 一 个 同名 的 .class 文 件 中 )。 为 了 生成 这 个 类 
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的 对 象 ， 运 行 这 个 程序 的 Java 虚 拟 机 (JVM) 将 使 用 被 称 为 “类 加 载 器 ”的 子 系统 。 

类 加 载 器 子 系统 实际 上 可 以 包含 一 条 类 加 载 器 链 ， 但 是 只 有 一 个 原生 类 加 载 器 ， 它 是 JVM 
实现 的 一 部 分 。 原 生 类 加 载 器 加 载 的 是 所 谓 的 可 信 类 ， 包 括 Java API 类 ， 它 们 通常 是 从 本 地 盘 加 
载 的 。 在 这 条 链 中 ， 通 常 不 需要 添加 额外 的 类 加 载 器 ， 但 是 如 果 你 有 特殊 需求 〈 例 如 以 某 种 特 
殊 的 方式 加 载 类 ， 以 支持 Web 服 务 器 应 用 ， 或 者 在 网 络 中 下 载 类 ) ， 那 么 你 有 一 种 方式 可 以 挂 接 
额外 的 类 加 载 器 。 

所 有 的 类 都 是 在 对 其 第 一 次 使 用 时 ， 动 态 加 载 到 JVM 中 的 。 当 程序 创建 第 一 个 对 类 的 静态 
成 员 的 引用 时 ， 就 会 加 载 这 个 类 。 这 个 证 明 构造 器 也 是 类 的 静态 方法 ， 即 使 在 构造 器 之 前 并 没 
有 使 用 static 关 键 字 。 因 此 ， 使 用 new 操 作 符 创建 类 的 新 对 象 也 会 被 当 作 对 类 的 静态 成 员 的 引用 。 

因此 ，Java 程 序 在 它 开始 运行 之 前 并 非 被 完全 加 载 ， 其 各 个 部 分 是 在 必需 时 才 加 载 的 。 这 
一 点 与 许多 传统 语言 都 不 同 。 动 态 加 载 使 能 的 行为 ， 在 诸如 C++ 这 样 的 静态 加 载 语言 中 是 很 难 
或 者 根本 不 可 能 复制 的 。 

类 加 载 器 首先 检查 这 个 类 的 Class 对 象 是 否 已 经 加 载 。 如 果 尚 未 加 载 ， 默 认 的 类 加 载 器 就 会 
根据 类 名 查找 ,class 文件 〈 例 如 ， 某 个 附加 类 加 载 器 可 能 会 在 数据 库 中 查找 字 节 码 ) 。 在 这 个 类 的 
字 节 码 被 加 载 时 ， 它 们 会 接受 验证 ， 以 确保 其 没有 被 破坏 ， 并 且 不 包含 不 良 Java 代 码 (这 是 Java 
中 用 于 安全 防范 目的 的 措施 之 一 ) 。 

一 旦 某 个 类 的 Class 对 象 被 载 人 内 存 ， 它 就 被 用 来 创建 这 个 类 的 所 有 对 象 。 下 面 的 示范 程序 
可 以 证 明 这 一 点 : 


/1/1: typeinfo/SweetShop.java 
// Examination of the way the class loader works. 
import static net.mindview.util.Print.*; 


class Candy { 
static { print("Loading Candy"); } 
) 


class Gum { 
static { print("Loading Gum"); } 


} 


class Cookie { 
static { print("Loading Cookie"): } 
} 


public class SweetShop { 
public static void main(String[] args) { 

print("inside main"); 

new Candy(); 

print("After creating Candy"); 

try { 
Class. forName ("Gum") ; 

} catch(ClassNotFoundException e) { 
print("Couldn't find Gum"); 


} 

print("After Class.forName(\"Gum\")"); 
new Cookie(); 

print("After creating Cookie"): 


} 
} /* Output: 
inside main 
Loading Candy 
After creating Candy 
Loading Gum 
After Class. forName("Gum") 
Loading Cookie 
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After creating Cookie 


这 里 的 每 个 类 Candy、Gum 和 Cookie， 都 有 一 个 static 子 句 ， 该 子 句 在 类 第 一 次 被 加 载 时 执 
行 。 这 时 会 有 相应 的 信息 打印 出 来 ， 告 诉 我 们 这 个 类 什么 时 候 被 加 载 了 。 在 main0 中 ， 创 建 对 
象 的 代码 被 置 于 打印 语句 之 间 ， 以 帮助 我 们 判断 加 载 的 时 间 点 。 

从 输出 中 可 以 看 到 ，Class 对 象 仅 在 需要 的 时 候 才 被 加 载 ，static 初 始 化 是 在 类 加 载 时 进行 的 。 

特别 有 趣 的 一 行 是 : 

Class.forName ("Gum"); 

这 个 方法 是 Class 类 (所 有 Class 对 象 都 属于 这 个 类 ) 的 一 个 static 成 员 。Class 对 象 就 和 其 他 
对 象 一 样 ， 我 们 可 以 获取 并 操作 它 的 引用 (这 也 就 是 类 加 载 器 的 工作 )。forName0 是 取得 Class 
对 象 的 引用 的 一 种 方法 。 它 是 用 一 个 包含 目标 类 的 文本 名 (注意 拼写 和 大 小 写 ) 的 String 作 输入 
参数 ， 返 回 的 是 一 个 Class 对 象 的 引用 ， 上 面 的 代码 忽略 了 返回 值 。 对 forName0 的 调用 是 为 了 它 
产生 的 “副作用 ": 如 果 类 Gum 还 没有 被 加 载 就 加 载 它 。 在 加 载 的 过 程 中 ，Gum 的 static 子 句 被 
执行 。 

在 前 面 的 例子 里 ， 如 果 Class.forName0 找 不 到 你 要 加 载 的 类 ， 它 会 抛 出 异常 ClassNot- 
FoundException。 这 里 我 们 只 需 简单 报告 问题 但 在 更 严密 的 程序 里 ， 可 能 要 在 异常 处 理 程序 
中 解决 这 个 问题 。 

无 论 何 时 ， 只 要 你 想 在 运行 时 使 用 类 型 信息 ， 就 必须 首先 获得 对 恰当 的 Class 对 象 的 引用 。 
Class.forName(0 就 是 实现 此 功能 的 便捷 途径 ， 因 为 你 不 需要 为 了 获得 Class 引 用 而 持 有 该 类 型 的 
对 象 。 但 是 ， 如 果 你 已 经 拥有 了 一 个 感 兴趣 的 类 型 的 对 象 ， 那 就 可 以 通过 调用 getClass( 方 法 来 
获取 Class 引 用 了 ， 这 个 方法 属于 根 类 Object 的 一 部 分 ， 它 将 返回 表示 该 对 象 的 实际 类 型 的 Class 
引用 。Class 包 含 很 多 有 用 的 方法 ， 下 面 是 其 中 的 一 部 分 : 


//: typeinfo/toys/ToyTest. java 


// Testing class Class. 
package typeinfo. toys; E 
import static net.mindview.util.Print.*; 


interface HasBatteries (} 
interface Waterproof {} 
interface Shoots {} 


class Toy { 
// Comment out the following default constructor 
// to see NoSuchMethodError from (*1*) 
Toy() {} z 
Toy(int i) () 

) 


class FancyToy extends Toy 

implements HasBatteries. Waterproof, Shoots { 
FancyToy() { super(1); } 

3 


public class ToyTest { 
static void printInfo(Class cc) { 
print("Class name: " + cc.getName() + 
" is interface? [" + cc.isInterface() + "]"); 
print("Simple name: ”+ cc.getSimpleName()); 
print("Canonical name : ”+ cc.getCanonicalName()); 


} 
public static void main(String{(] args) { 
Class ¢ = null; 
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public static void main(String[] args) {, 
Class c = null; 
try { 
c = Class. forName("typeinfo. toys. FancyToy"); 
} catch(ClassNotFoundException e) { 
print("Can't find FancyToy"); m 
System.exit(1); 


} 

printInfo(c); 

for(Class face : c.getInterfaces()) 
printInfo(face) ; 

Class up = c.getSuperclass(); 

Object’ obj = nutt; 

try { 

// Requires default constructor: 

obj = up.newInstance() ; 

catch(InstantiationException e) { 

print("Cannot instantiate"): . 

System.exit(1); 

catch(IlLegalaccessException e) { 

print("Cannot access"); 

System.exit(1) 5 


} 
printInfo(obj .getClass()); 


} 
} /* Output: 
Class name: typeinfo.toys.FancyToy is interface? [false] 
Simple name: FancyToy 
Canonical name : typeinfo.toys.FancyToy 
Class name: typeinfo. toys.HasBatteries îs interface? [true] 
Simple name: HasBatteries 
Canonical name : typeinfo.toys.HasBatteries 
Class name: typeinfo.toys.Waterproof is interface? [true] 
Simple name: Waterproof 
Canonical name : typeinfo.toys.Waterproof i 
Class name: typeinfo.toys.Shoots is interface? [true] 
Simple name: Shoots . 
Canonical name : typeinfo. toys. Shoots 
Class name: typeinfo.toys.Toy is interface? [false] 
Simple name: Toy 
Canonical name : typeinfo. toys. Toy 
:~ 


FancyToy 继 承 自 Toy 并 实现 了 HasBatteries、Waterproof 和 Shoots 接 口 。 在 main0 中 ， 用 
forName() 方 法 在 适当 的 try 语 句 块 中 ， 创 建 了 一 个 Class 引 用 ， 并 将 其 初始 化 为 指向 FancyToy 
Class。 注 意 ， 在 传递 给 forName0 的 字符 串 中 ， 你 必须 使 用 全 限定 名 (包含 包 名 )。 

printInfo() 使 用 getrName() 来 产生 全 限定 的 类 名 ， 并 分 别 使 用 getSimpleName() 和 
getCanonicalName() (在 Java SE5 中 引入 的 ) 来 产生 不 含 包 名 的 类 名 和 全 限定 的 类 名 。 
isInterface() 方 法 如 同 其 名 ， 可 以 告诉 你 这 个 Class 对 象 是 否 表示 某 个 接口 。 因 此 ， 通 过 Class 对 
象 ， 你 可 以 发 现 你 想 要 了 解 的 类 型 的 所 有 信息 。 

在 main0 中 调用 的 Class.getInterfaces() 方 法 返回 的 是 Class 对 象 ， 它 们 表示 在 感 兴趣 的 Class 
对 象 中 所 包含 的 接口 。 ， 

如 果 你 有 一 个 Class 对 象 ， 还 可 以 使 用 getSuperclass0 方 法 查询 其 直接 基 类 ， 这 将 返回 你 可 以 
用 来 进一步 查询 的 Class 对 象 。 因 此 ， 你 可 以 在 运行 时 发 现 一 个 对 象 完整 的 类 继承 结构 。 

Class 的 newInstance() 方 法 是 实现 “虚拟 构造 器 ”的 一 种 途径 ， 虚 拟 构造 器 允许 你 声明 ， 
“我 不 知道 你 的 确切 类 型 ， 但 是 无 论 如 何 要 正确 地 创建 你 自己 。” 在 前 面 的 示例 中 ，up 仅 仅 只 是 
一 个 Class 引 用 ， 在 编译 期 不 具备 任何 更 进一步 的 类 型 信息 。 当 你 创建 新 实例 时 ， 会 得 到 Object 
引用 ， 但 是 这 个 引用 指向 的 是 Toy 对 象 。 当 然 ， 在 你 可 以 发 送 Object 能 够 接受 的 消息 之 外 的 任何 
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消息 之 前 ， 你 必须 更 多 地 了 解 它 ， 并 执行 某 种 转型 。 另 外 ， 使 用 newInstance0 来 创建 的 类 ， 必 
须 带 有 默认 的 构造 器 。 在 本 章 稍 后 部 分 ， 你 理会 有 到 如 何 通过 合用 mm 的 民间 AP 用 任意 的 构 
造 器 来 动态 地 创建 类 的 对 象 。 

练习 1: (1) 在 ToyTestjava 中 ， 将 Toy 的 默认 构造 器 注释 掉 ， JMR. 

练习 2: (2) 将 新 的 interface 加 到 ToyTestjava 中 ， 看 看 这 个 程序 是 否 能 够 正确 检测 出 来 并 加 
以 显示 。 

练习 3: (2) 将 Rhomboid (27%) 加 入 Shapes.java 中 。 创 建 一 个 Rhomboid， 将 其 向 上 转型 
为 Shape， 然 后 向 下 转型 回 Rhomboid。 试 着 将 其 向 下 转型 成 Circle， 看 看 会 发 生 什 么 。 

练习 4: (2) 修改 前 一 个 练习 ， 让 你 的 程序 在 执行 向 下 转型 之 前 先 运用 instanceof 检 查 类 型 。 

练习 5: (3) 实现 Shapes.java 中 的 rotate(Shape) 方 法 ， 让 它 能 判断 它 所 旋转 的 是 不 是 Circle 
(如 果 是 ， 就 不 执行 )。 

练习 6: (4) 修改 Shapes.java， 使 这 个 程序 能 将 某 个 特定 类 型 的 所 有 形状 都 “标示 ”出 来 
(通过 设 标志 )。 每 一 个 导出 的 Shape 类 的 toString0 方 法 应 该 更 够 指出 Shape 是 否 被 标示 。 

练习 7: (3) 修 改 SweetShopsjava， 使 每 种 类 型 对 象 的 创建 由 命令 行 参数 控制 。 即 ， 如 果 命令 
行 是 “java SweetShop Candy"， 那 么 只 有 Candy 对 象 被 创建 。 注 意 你 是 如 何 通过 命令 行 参数 来 
控制 加 载 哪个 Class 对 象 的 。 

练习 8: (5) 写 一 个 方法 ， 令 它 接受 任意 对 象 作为 参数 ， 并 能 够 递归 打印 出 该 对 象 所 在 的 继 
承 体系 中 的 所 有 类 。 

练习 9: (5) 修改 前 一 个 练习 ， 让 这 个 方法 使 用 Class.getDeclaredFields0 来 打印 一 个 类 中 的 
域 的 相关 信息 。 

练习 10: (3) 写 一 个 程序 ， 使 它 能 判断 char 数 组 究竟 是 个 基本 类 型 ， 还 是 一 个 对 象 。 
14.2.1 类 字面 常量 

Java 还 提供 了 另 一 种 方法 来 生成 对 Class 对 象 的 引用 ， 即 使 用 类 字面 常量 。 对 上 述 程 序 来 说 ， 
就 像 下 面 这 样 : 

FancyToy.class; 
这 样 做 不 仅 更 简单 ， 而 且 更 安全 ,因为 它 在 编译 时 就 会 受到 检查 (因此 不 需要 置 于 try 语 句 块 中 ) 。 
并 且 它 根除 了 对 forName0 方 法 的 调用 ， 所 以 也 更 高 效 。 ，! 

类 字面 常量 不 仅 可 以 应 用 于 普通 的 类 ， 也 可 以 应 用 于 接口 、 数 组 以 及 基本 数据 类 型 。 另 外 ， 
对 于 基本 数据 类 型 的 包装 器 类 ， 还 有 一 个 标准 字段 TYPE。TYPE 字 段 是 一 个 引用 ， 指 向 对 应 的 
基本 数据 类 型 的 Class 对 象 ， 如 下 所 示 : 
































类 型 信息 319 





我 建议 使 用 “.class” 的 形式 ， 以 保持 与 普通 类 的 一 致 性 。 
注意 ， 有 一 点 很 有 趣 ， 当 使 用 “.class” 来 创建 对 Class 对 象 的 引用 时 ， 不 会 自动 地 初始 化 该 


Class 对 象 。 为 了 使 用 类 而 做 的 准备 工作 实际 包含 三 个 步骤 : 

1. 加载 ， 这 是 由 类 加 载 器 执行 的 。 该 步 又 将 查找 字 节 码 (通常 在 classpath 所 指定 的 路 径 中 查 
找 ， 但 这 并 非 是 必需 的 ) ， 并 从 这 些 字 节 码 中 创建 一 个 Class 对 象 。 

2. 链接 。 在 链接 阶段 将 验证 类 中 的 字 节 码 ， 为 静态 域 分 配 存储 空间 ， 并 且 如 果 必 需 的 话 ， 


将 解析 这 个 类 创建 的 对 其 他 类 的 所 有 引用 。 
3. 初始 化 。 如 果 该 类 具有 超 类 ， 则 对 其 初始 化 ， 执 行 静态 初始 化 器 和 静态 初始 化 块 。 
初始 化 被 延迟 到 了 对 静态 方法 (构造 器 隐 式 地 是 静态 的 ) 或 者 非常 数 静态 域 进行 首次 引用 


时 才 执 行 : 


//: typeinfo/ClassInitialization. java 
import java.util.*; 


class Initable { 
static final int staticFinal = 47; 
static final int staticFinal2 = 
ClassInitialization. rand. nextInt (1088) ; 
static { 
System.out.printin("Initializing Initable"); 
} 
} 


class Initable2 { 
Static int staticNonFinal = 147; 


static { 
System.out.printin("Initializing Initable2"); 


} 


class Initable3 { 
static int staticNonFinal = 74; 


static { 
System.out.printin("Initializing Initable3"); 


} 
) 


public class ClassInitialization { 
public static Random rand = new Random(47); 
public static void main(String{] args) throws Exception { 
Class initable = Initable.class; 
System.out.printin("After creating Initable ref"): 
1 // Does not trigger initialization: 
System.out.printIn(Initable. stat icFinal); 
// Does trigger initialization: 
System. out.printin(Initable. staticFinal2) ; 
// Does trigger initialization: 
System.out.printin(Initable2.staticNonFinat 
Class initable3 = Class. forName("Initable3” 
System.out.printin("After creating Initable3 ref"); 
System. out.printIn(Initable3.staticNonFinal); 











} 
} /* Output: 
After creating Initable ref 
47 
Initializing Initable 
258 


Initializing Initable2 
147 
Initializing Initable3 

After creating Initable3 ref 
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初始 化 有 效 地 实现 了 尽 可 能 的 “ 情 性 ”"。 从 对 initable 引 用 的 创建 中 可 以 看 到 ， 仅 使 用 .class 
语法 来 获得 对 类 的 引用 不 会 引发 初始 化 。 但 是 , 为 了 产生 Class 引 用 ，Cilass.forNameO 立 即 就 进 
行 了 初始 化 ， 就 像 在 对 initable3 引 用 的 创建 中 所 看 到 的 。 

如 果 一 个 static final 值 是 “编译 期 常量 "， 就 像 Initable.staticFinal 那 样 ， 那 么 这 个 值 不 需要 
对 Initable 类 进行 初始 化 就 可 以 被 读 取 。 但 是 ， 如 果 只 是 将 一 个 域 设置 为 static 和 final 的 ， 还 不 足 
以 确保 这 种 行为 ， 例 如 ， 对 Initable.staticFinal2 的 访问 将 强制 进行 类 的 初始 化 ， 因 为 它 不 是 一 个 
编译 期 常量 。 

”如 果 一 个 static 域 不 是 final 的 ， 那 么 在 对 它 访问 时 ， 总 是 要 求 在 它 被 读 取 之 前 ， 要 先进 行 链 
接 (为 这 个 域 分 配 存储 空间 ) 和 初始 化 (初始 化 该 存储 空间 ) ， 就 像 在 对 Initable2.staticNonFinal 
的 访问 中 所 看 到 的 那样 。 

14.2.2 泛 化 的 Class 引 用 

Class 引 用 总 是 指向 某 个 Class 对 象 ， 它 可 以 制造 类 的 实例 ， 并 包含 可 作用 于 这 些 实例 的 所 有 
方法 代码 。 它 还 包含 该 类 的 静态 成 员 ， 因 此 ，Class 引 用 表示 的 就 是 它 所 指向 的 对 象 的 确切 类 型 ， 
而 该 对 象 便 是 Class 类 的 一 个 对 象 。 

但 是 ，Java SE5 的 设计 者 们 看 准 机 会 ， 将 它 的 类 型 变 得 更 具体 了 一 些 ， 而 这 是 通过 允许 你 对 
Class 引 用 所 指向 的 Class 对 象 的 类 型 进行 限定 而 实现 的 , 这 里 用 到 了 泛 型 语法 。 在 下 面 的 实例 中 ， 
两 种 语法 都 是 正确 的 : 


//: typeinfo/GenericClassReferences. java 


public class GenericClassReferences { 
public static void main(String[] args) { 
Class intClass = int.class; 
Class<Integer> genericIntClass = int.class; 
genericIntClass = Integer.class; // Same thing 
intClass = double.class; 
// genericIntClass = double.class: // Illegal 


} 

} M~ 

普通 的 类 引用 不 会 产生 警告 信息 ， 你 可 以 看 到 ， 尽 管 泛 型 类 引用 只 能 赋值 为 指向 其 声明 的 
类 型 ， 但 是 普通 的 类 引用 可 以 被 重新 赋值 为 指向 任何 其 他 的 Class 对 象 。 通 过 使 用 泛 型 语法 ， 可 
以 让 编译 器 强制 执行 额外 的 类 型 检查 。 

如 果 你 希望 稍微 放松 一 些 这 种 限制 ， 应 该 怎么 办 呢 ? 乍 一 看 ， 和 好像 你 应 该 能 够 执行 类 似 下 
面 这 样 的 操作 : 

Class<Number> genericNumberClass = int.class; 
这 看 起 来 似乎 是 起 作用 的 ， 因 为 Integer 继 承 自 Number。 但 是 它 无 法 工作 ， 因 为 Integer Class 
对 象 不 是 Number Class 对 象 的 子 类 ARERIA 我 们 将 在 第 15 章 中 深入 讨 
WE). 

为 了 在 使 用 泛 化 的 Class 引 用 时 放松 限制 ， 我 使 用 了 通配符 ， 它 是 Java 泛 型 的 一 部 分 。 通 配 
符 就 是 “?”"， 表 示 “ 任 何事 物 "。 因 此 ， 我 们 可 以 在 上 例 的 普通 Class 引 用 中 添加 通配符 ， 并 产 
生 相同 的 结果 ; 


//: typeinfo/WildcardClassReferences. java 


Public class WildcardClassReferences { s 
public static void main(String[] args) { i A 


[km ~ o— Ser, © 加 ~~ = 
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Class<?> intClass = int.class; ` w 
intClass = double.class; 


eae 


在 Java SES, Ciass<?> 优 于 平凡 的 Class， 即便 它们 是 等 价 的 ， 并 目下 及 的 Cia 各 你 所 见 ， 
不 会 产生 编译 器 警告 信息 。Class<?> 的 好 处 是 它 表 示 你 并 非 是 碰巧 或 者 由 于 疏忽 ， 而 使 用 了 一 
个 非 具体 的 类 引用 ， 你 就 是 选择 了 非 具 体 的 版 本 。 

为 了 创建 一 个 Class 引 用 ， 它 被 限定 为 某 种 类 型 ， 或 该 关 型 的 任何 子 类 型 ， 你 需要 将 通配符 
与 extends 关 键 字 相 结合 ， 创 建 一 个 范围 。 因 此 ， 与 仅仅 声明 Class<Number> 不 同 ， 现 在 做 如 下 
声明 : ; s 


//: typeinfo/BoundedClassReferences. java 


public class BoundedClassReferences { ` 
public static void main(String[] args) { 
Class<? extends Number> bounded = int.class; 
bounded = double.class; 
bounded = Number .class; 
// Or anything else derived from Number ș 


} 
dM ~ 


向 Class 引 用 添加 泛 型 语法 的 原因 仅仅 是 为 了 提供 编译 期 类 型 检查 ， 因 此 如 果 你 操作 有 误 ; 
稍 后 立即 就 会 发 现 这 一 点 。 在 使 用 普通 Class 引 用 ， 你 不 会 误 和 歧途， 但 是 如 果 你 确实 犯 了 错误 ， 
那么 直到 运行 时 你 才 会 发 现 它 ， 而 这 显得 很 不 方便 。 

下 面 的 示例 使 用 了 泛 型 类 语法 。 它 存储 了 一 个 类 引用 ， 稍 候 又 产生 了 一 个 List， 填充 这 个 
List 的 对 象 是 使 用 newInstance0 方 法 ， 通 过 该 引用 生成 的 : 


//: typeinfo/FilledList. java 
import java.util. *; 


X 


class CountedInteger { 
private static long counter: 
private final long id = counter++; 
public String toString() { return Long.toString(id); } 
} 
be] 
public class FilledList<T> { 
private Class<T> type; 
public FilledList(Class<T> type) { this.type = type; } 
public List<T> create(int nElements) { 
ListeT> result = new ArrayList<T>(); 
try { 
for(int 1 = 0; 1 < nElements; i++) -a 
result.add(type.newInstance()):; f “ 
} cateh(Exception e) { 
throw new RuntimeException(e); . 


} 
return result; 


public static void main(String{] args) { 
Fiiledtist<CountedInteger> fl = 
new FilledList<CountedInteger>(CountedInteger.class) ; 
System.out.printin(fl.create(15)); 


} ~ 
} /* Output: he 
(8, 1, 2, 3, 4, 5, 6, 7, 8, 9, 16, 11, 12, 13, 14) 
i~ 


注意 ， 这 个 类 必须 假设 与 它 一 同 工 作 的 任何 类 型 都 具有 一 个 默认 的 构造 器 (无 参 构造 器 )， 
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并 且 如 果 不 符 合 该 条 件 ， 你 将 得 到 一 个 异常 。 编 译 器 对 该 程序 不 会 产生 任何 警告 信息 。 
当 你 将 泛 型 语法 用 于 Class 对 象 是 ， 会 发 生 一 件 很 有 趣 的 事情 : newInstance0 将 返回 该 对 象 
的 确切 类 型 ， 而 不 仅仅 只 是 在 ToyTestjava 中 看 到 的 基本 的 Object。 这 在 某 种 程度 上 有 些 受 限 : 


//: typeinfo/toys/GenericToyTest.java 

/1 Testing class Class. 

package typeinfo. toys; 

public class GenericToyTest { 

Public static void main(Stringl] args) throws Exception { 

Class<FancyToy> ftClass = FancyToy.class; 
// Produces exact type: 
FancyToy fancyToy = ftClass.newInstance(); 
Class<? super FancyToy> up = ftClass.getSuperclass(); 
// This won't compile: 
// Class<Toy> up2 = ftClass.getSuperclass(): 
// Only produces Object: 
Object obj = up.newInstance(): 


} 

} Ii~ 

如 果 你 手头 的 是 超 类 ， 那 编译 器 将 只 允许 你 声明 超 类 引用 是 “ 某 个 类 ， 它 是 FancyToy 超 类 ”， 
就 像 在 表达 式 Class<? Super FancyToy> 中 所 看 到 的 ， 而 不 会 接受 Class<Toy> 这 样 的 声明 。 这 看 
上 去 显得 有 些 怪 ， 因 为 getSuperClass0 方 法 返回 的 是 基 类 (不 是 接口 )， 并 且 编 译 器 在 编译 期 就 
知道 它 是 什么 类 型 了 一 一 在 本 例 中 就 是 Toy.class 一 一 而 不 仅仅 只 是 “ 某 个 类 , 它 是 FancyToy 超 类 ”。 
不 管 怎样 ， 正 是 由 于 这 种 含糊 性 ，up.newInstance0 的 返回 值 不 是 精确 类 型 ， 而 只 是 Object。 
14.2.3 新 的 转型 语法 

* Java SES 还 添加 了 用 于 Class 引 用 的 转型 语法 ， 即 cast0 方 法 : 


//: typeinfo/ClassCasts. java 


class Building {} 
class House extends Building {} 


public class ClassCasts { 
public static void main(String[] args) { 
Building b = new House(); 
Class<House> houseType = House.class; 
House h = houseType.cast(b); 
h = (House)b; // ... or just do this. 


} 

} Mi~ . 

cast0 方 法 接受 参数 对 象 ， 并 将 其 转型 为 Class 引 用 的 类 型 。 当 然 ， 如 果 你 观察 上 面 的 代码 ， 
则 会 发 现 ， 与 实现 了 相同 功能 的 main0 中 最 后 一 行 相 比 ， 这 种 转型 好 像 做 了 很 多 额外 的 工作 。 
新 的 转型 语法 对 于 无 法 使 用 普通 转型 的 情况 显得 非常 有 用 ， 在 你 编写 泛 型 代码 (你 将 在 第 15 章 
中 学 习 它 ) 时 ， 如 果 你 存储 了 Class 引 用 ， 并 希望 以 后 通过 这 个 引用 来 执行 转型 ， 这 种 情况 就 会 
时 有 发 生 。 这 被 证 明 是 一 种 罕见 的 情况 一 我 发 现在 整个 Java SE5 类 库 中 ， 只 有 一 处 使 用 了 
cast() (在 com.sun.mirror.util.DeclarationFilter 中 )。 

在 Java SE5 中 另 一 个 没有 任何 用 处 的 新 特性 就 是 Class.asSubclass0， 该 方法 允许 你 将 一 个 类 
对 象 转型 为 更 加 具体 的 类 型 。 


14.3 类 型 转换 前 先 做 检查 


迄今 为 止 ， 我 们 已 知 的 RTTI 形 式 包括 : 
D 传统 的 类 型 转换 ， 如 “(Shape)”， 由 RTTI 确 保 类 型 转换 的 正确 性 ， 如 果 执行 了 一 个 错误 
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的 类 型 转换 ， 就 会 抛 出 一 个 ClassCastException 异 常 。 

D) 代表 对 象 的 类 型 的 Class 对 象 。 通 过 查询 Class 对 象 可 以 获取 运行 时 所 需 的 信息 。 

在 C++ 中 ， 经 典 的 类 型 转换 “(Shape)” 并 不 使 用 RTTI。 它 只 是 简单 地 告诉 编译 器 将 这 个 对 
象 作为 新 的 类 型 对 待 。 而 Java 要 执行 类 型 检查 ， 这 通常 被 称 为 “类 型 安全 的 向 下 转型 "。 之 所 以 
叫 “ 向 下 转型 "， 是 由 于 类 层次 结构 图 从 来 就 是 这 么 排列 的 。 如 果 将 Cirele 类 型 转换 为 Shape 类 型 
被 称 作 向 上 转型 ， 那 么 将 Shape 转 型 为 Circle， 就 被 称 为 向 下 转型 。 但 是 ， 由 于 知道 Circle 肯 定 
是 一 个 Shape， 所 以 编译 器 允许 自由 地 做 向 上 转型 的 赋值 操作 ， 而 不 需要 任何 显 式 的 转型 操作 。 
编译 器 无 法 知道 对 于 给 定 的 Shape 到 底 是 什么 Shape 一 一 它 可 能 就 是 Shape, 或 者 是 Shape 的 字 类 型 ， 
例如 Circle、Square、Triangle 或 某 种 其 他 的 类 型 。 在 编译 期 。 编 译 器 只 能 知道 它 是 Shape。 因 此 ， 
如 果 不 使 用 显 式 的 类 型 转换 ， 编 译 器 就 不 允许 你 执行 向 下 转型 赋值 ， 以 告知 编译 器 你 拥有 额外 
的 信息 ， 这 些 信息 使 你 知道 该 类 型 是 某 种 特定 类 型 (编译 器 将 检查 向 下 转型 是 否 合理 ， 因 此 它 
不 允许 向 下 转型 到 实际 上 不 是 待 转型 类 的 子 类 的 类 型 上 ) 。 

RTTI 在 Java 中 还 有 第 三 种 形式 ， 就 是 关键 字 instanceof。 它 返回 一 个 布尔 值 ， 告 诉 我 们 对 象 
是 不 是 某 个 特定 类 型 的 实例 。 可 以 用 提问 的 方式 使 用 它 ， 就 像 这 样 : 


if(x instanceof Dog) 
((Dog)x) .bark() ; 


在 将 x 转 型 成 一 个 Dog 前 ， 上 面 的 这 语句 会 检查 对 象 xz 是 否 从 属于 Dog 类 。 进 行 向 下 转型 前 ， 
如 果 没 有 其 他 信息 可 以 告诉 你 这 个 对 象 是 什么 类 型 ， 那 么 使 用 instanceof 是 非常 重要 的 ， 否 则 会 
得 到 一 个 ClassCastException 异 常 。 

一 般 ， 可 能 想 要 查找 某 种 类 型 (比如 要 找 三 角形 ， 并 填充 成 紫色 ) ， 这 时 可 以 轻松 地 使 用 
instanceof 来 计数 所 有 对 象 。 例 如 ， 假 设 你 有 一 个 类 的 继承 体系 ， 描 述 了 Pet (以 及 它们 的 主人 ， 
这 是 在 后 面 的 示例 中 出 现 的 一 个 非常 方便 的 特性 )。 在 这 个 继承 体系 中 的 每 个 Individual 都 有 一 
个 id 和 一 个 可 选 的 名 字 。 尽 管 下 面 的 类 都 继承 自 Individual， 但 是 Individual 类 复杂 性 较 高 ， 因 此 
其 代码 将 放 到 第 17 章 中 进行 说 明 与 解释 。 正 如 你 可 以 看 到 的 ， 此 处 并 不 需要 去 了 解 Individual 的 
代码 一 -你 只 需 了 解 你 可 以 创建 其 具名 或 不 具名 的 对 象 ， 并 且 每 个 Individual 都 有 一 个 id0 方 法 ， 
可 以 返回 其 唯一 的 标识 符 (通过 对 每 个 对 象 计数 而 创建 的 )。 还 有 一 个 toString0 方 法 ， 如 果 你 没 
有 为 Individual 提 供 名 字 ，toString0 方 法 只 产生 类 型 名 。 

下 面 是 继承 自 Individual 的 类 继承 体系 : 


//: typeinfo/pets/Person. java 
package typeinfo.pets; 





public class Person extends Individual { 
Public Person(String name) { super(name); } 
} Mi~ 


//: typeinfo/pets/Pet. java 
package typeinfo.pets; 


public class Pet extends Individual { 
public Pet(String name) { super(name); } 
public Pet() { super(); } 

} Mi~ 


//: typeinfo/pets/Dog. java 
package typeinfo.pets; 
public class Dog extends Pet { 

Public Dog(String name) { super(name); } 


public Dog() { super(); } 
} M~ 
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//: typeinfo/pets/Mutt. java 
package typeinfo.pets; 


public class Mutt extends Dog { 
public Mutt(String name) { super(name); } 
public Mutt() { super(); } 

} M~ 


71: typeinfo/pets/Pug. java 
package typeinfo.pets! 


public’ class Pug extends Dog { 
public Pug(String name) { super(name); } 
public Pug() { super(); } 

} M~ 


//: typeinfo/pets/Cat. java 
package typeinfo.pets: 


public class Cat extends Pet { 
public Cat(String name) { super(name); } 
public Cat() { super(); } 

} = 


//: typeinfo/pets/Egyptianmau. java 
package typeinfo.pets; 


public class EgyptianMau extends Cat { 
public EgyptianMau(String name) { super(name); } 
public EgyptianMau() { super(); } 

} Mi~ 


//: typeinfo/pets/Manx. java 
package typeinfo.pets; 


public class Manx extends Cat { 
public Manx (String name) { super (name); 
public Manx() { super(); } 

} Mi~ : í à 


//: typeinfo/pets/Cymric. java 
package typeinfo.pets: , 





public class Cymric extends Manx { 
public Cymric(String name) { super(name); } 
public Cymric() { super(); } 

y Mi~ 


//: typeinfo/pets/Rodent. java 
package typeinfo.pets; 


public class Rodent extends Pet { 
Public Rodent (String name) { super(name); } 
Public Rodent() { super(); } 

} Mi~ 


/1: typeinfo/pets/Rat. java 
package typeinfo.pets; 


public class Rat extends Rodent { 
public Rat(String name) { super(name); } 
public Rat() { super(); } 

YUH 


//: typeinto/pets/Mouse. java 
package typeinfo.pets; 


Public class Mouse extends Rodent { 
public Mouse(String name) { super(name); } 
public Mouse() { super(); } 





类 型 信息 325 





} AAA 


//: typeinfo/pets/Hamster.java 
package typeinfo.pets; 


public class Hamster extends Rodent { 
public Hamster (String name) { super(name); } 
public Hamster() { super(); } 

} Mi~ 


接 下 来 ， 我 们 需要 一 种 方法 ， 通 过 它 可 以 随机 地 创建 不 同类 型 的 宠物 ， 并 且 为 方便 起 见 ， 
还 可 以 创建 宠物 数组 和 List。 为 了 使 该 工具 能 够 适应 多 种 不 同 的 实现 ， 我 们 将 其 定义 为 抽象 类 ; 

JJ: typeinfo/pets/PetCreator. java 

// Creates random sequences of Pets. F 


paċkage typeinfo.pets; 
import java.util.*; 











[572] 





public abstract class PetCreator { 
private Random rand = new Random(47).; 
// The List of the different types of Pet to create: 
public abstract List<Class<? extends Pet>> types(); 
public Pet randomPet() { // Create one random Pet 
int n = rand.nextInt(types().size()): 
try { 
return types().get(n).newInstance(); 
} catch(InstantiationException e) { 
throw new RuntimeException(e) ; 
} catch(IllegalAccessException e) { ` 
throw new Runt imeException(e) ; 


} 


} 

Public Pet[] createArray(int size) ¢ 
Pet[] result = new Pet[size]; 
for(int i = 9; i < size; i++) 

result[1] = randomPet(); 
return result; 

} 


Public ArrayList<Pet> arrayList(int size) { 
ArrayList<Pet> result = new ArrayList<Pet>(); ra 
Collections.addAll (result, createArray(size)); 
return result; 

} 

} /AL 

抽象 的 getTypes0 方 法 在 导出 类 中 实现 ， 以 获取 由 Class 对 象 构 成 的 List (这 是 模版 方法 设计 
模式 的 一 种 变 体 )。 注 意 ， 其 中 类 的 类 型 被 指定 为 “任何 从 Pet 导 出 的 类 ”"， 因 此 newInstanceO 不 
需要 转型 就 可 以 产生 Pet。randomPet0 随 机 地 产生 List 中 的 索引 ， 并 使 用 被 选取 的 Class 对 象 ， 
通过 Class.newInstance0 来 生成 该 类 的 新 实例 。createArray0 方 法 使 用 randomPet0 来 填充 数组 ， 
而 arrayList0 方 法 使 用 的 则 是 createArray0。 

在 调用 newInstance0 时 ， 可 能 会 得 到 两 种 异常 ， 在 紧 跟 try 语 句 块 后 面 的 catch 子 句 中 可 以 看 
到 对 它们 的 处 理 。 异 常 的 名 字 再 次 成 为 了 一 种 对 错误 类 型 相对 比较 有 用 的 解释 (TllegalAccess- 
Exception 表 示 违 反 了 Java 安 全 机 制 ， 在 本 例 中 ， 表 示 默 认 构造 器 为 private 的 情况 )。 

当 你 导出 PetCreator 的 子 类 时 ， 唯 一 所 需 提供 的 就 是 你 希望 使 用 randomPet0 和 其 他 方法 来 
创建 的 宠物 类 型 的 List。getTypes() 方 法 通常 只 返回 对 一 个 静态 List 的 引用 。 下 面 是 使 用 
forName0 的 一 个 具体 实现 : 

//: typeinfo/pets/ForNameCreator.java 


package typeinfo.pets; 
import java.util.*; 
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public class ForNameCreator extends PetCreator { 
private static List<Class<? extends Pet>> types = ` G 
new ArrayList<Class<? extends Pet>>(); ‘ 
// Types that you want to be randomly created: 
private static String[] typeNames = { 
"typeinfo.pets.Mutt", 
“typeinfo.pets.Pug", 
"typeinfo. pets. EgyptianMau", 
“typeinfo.pets.Manx", 
“typeinfo. pets.Cymric", 
"typeinfo.pets.Rat", 
"typeinfo.pets.Mouse", 
“typeinfo.pets.Hamster” 
a 
@SuppressWarnings ("unchecked") 
Private static void loader() { 
try { 
for(String name : typeNames) 
types. add( 
(Class<? extends Pet>)Class. forName(name)):; 
} catch(ClassNotFoundException e) { 
throw new Runtime€xception(e) ; a ft 
} 
static { loader(); } 
public List<Class<? extends Pet>> types() {return types} 
YM hm 


1 loader() 方 法 用 Class.forName0 创 建 了 Class 对 象 的 List , 这 可 能 会 产生 ClassNotFound- 
H Exception 异 常 ， 这 么 做 是 有 意义 的 ， 因 为 你 传递 给 它 的 是 一 个 在 编译 期 无 法 验证 的 String。 由 
于 Pet 对 象 在 typeinfo 包 中 ， 因 此 必须 使 用 包 名 来 引用 这 些 类 。 
为 了 产生 具有 实际 类 型 的 Class 对 象 的 List， 必 须 使 用 转型 ， 这 会 产生 编译 期 警告 。loader0 
574] 方法 被 单独 定义 ， 然 后 被 置 于 一 个 静态 初始 化 子 句 中 ， 因 为 @SuppressWarnings 注 解 不 能 直接 
置 于 静态 初始 化 子 句 之 上 。 
为 了 对 Pet 进行 计数 ， 我 们 需要 一 个 能 够 跟踪 各 种 不 同类 型 的 Pet 的 数量 的 工具 。Map 是 此 
需求 的 首选 ， 其 中 刍 是 Pet 类 型 名 ， 而 值 是 保存 Pet 数量 的 integer。 通 过 这 种 方式， 你 可 以 询问 ， 
“有 多 少 个 Hamster 对 象 ? ”我 们 可 以 使 用 instanceof 来 对 Pet 进行 计数 ，“ 


/1: typeinfo/PetCount .java 
// Using instanceof. 
import typeinfo.pets.*; 
import java.util.*; 
import static net.mindview.util.Print.*; 

















public class PetCount, { 5 

static class PetCounter extends: HashMap<String,Integer> { 4 

public void count(String type) { 

Integer quantity = get (type); 
if(quantity == null) 

put(type, 1); 

etse ， 

put(type, quantity + 1); 









} . 
public static void 
countPets(PetCreator creator) { $ £ 
PetCounter counter= new PetCounter(); R p 
for(Pet pet : creator.createArray(20)) { 
// List each individual pet: 
Printnb(pet.getClass().getSimpleName() + “ 
if(pet instanceof Pet) 
counter .count ("Pet") ; 
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1f(pet instanceof Dog) 
counter .count ("Dog") > 
if(pet instanceof Mutt) 
counter .count ("Mutt"); _ 
if(pet instanceof Pug) 
counter „count ("Pug") ; 
if(pet instanceof Cat) 
counter .count("Cat") ; 
if(pet instanceof Manx) 
counter .count ("EgyptianMau") ; 
if(pet instanceof Manx) 
counter.count ("Manx") ; 
if(pet instanceof Manx) 
counter.count("Cymric"); 
if(pet instanceof Rodent) 
counter .count ("Rodent"); 
if(pet instanceof Rat) 
counter „count ("Rat") ; 
if(pet instanceof Mouse) 
counter .count ("Mouse"); 
if(pet instanceof Hamster) 
counter. count ("Hamster") ; 


} 

// Show the counts: 
print(); 
print(counter); 


u 
public static void main(String(] args) { 
countPets(new ForNameCreator()); 


} 
} /* Output: 
Rat Manx Cymric Mutt Pug Cymric Pug Manx Cymric Rat 
EgyptianMau Hamster EgyptianMau Mutt Mutt Cymric Mouse Pug 
Mouse Cymric 
{Pug=3, Cat=9, Hamster=1, Cymric=7, Mouse=2, Mutt=3, 
Rodent=5, Pet=20, Manx=7, EgyptianMau=7, Dog=6, Rat=2} 
Wh 


在 CountPets0 中 ， 是 使 用 PetCreator 来 随机 地 向 数组 中 填充 Pet 的 。 然 后 使 用 jinstanceof 对 该 
数组 中 的 每 个 Pet 进 行 测试 和 计数 。 

对 instanceof 有 比较 严格 的 限制 : 只 可 将 其 与 命名 类 型 进行 比较 , 而 不 能 与 Class 对 象 作 比 较 。 
在 前 面 的 例子 中 ， 可 能 觉得 写 出 那么 一 大 堆 instanceof 表 达 式 是 很 过 味 的 ， 的 确 如 此 。 但 是 也 没 
有 办 法 让 instanceof 聪 明 起 来 ， 让 它 能 够 自动 地 创建 一 个 Class 对 象 的 数组 ， 然 后 将 目标 对 象 与 这 
个 数组 中 的 对 象 进行 逐一 的 比较 〈 稍 后 会 看 到 一 个 替代 方案 )。 其 实 这 并 非 是 一 种 如 你 想象 中 那 
般 好 的 限制 ， 因 为 渐渐 地 读者 就 会 理解 ， 如 果 程 序 中 编写 了 许多 的 instanceof 表 达 式 ， 就 说 明 你 
的 设计 可 能 存在 瑕 症 。 


14.3.1 使 用 类 字面 常量 
如 果 我 们 用 类 字面 常量 重新 实现 PetCount， 那 么 改写 后 的 结果 在 许多 方面 都 会 显得 更 加 清 
Wi: 


//: typeinfo/pets/LiteralPetCreator. java 
// Using class literals. 

package typeinfo. pets; 

import java.util.*; 


public class LiteralPetCreator extends PetCreator { 
// No try block needed. 
@SuppressWarnings ("unchecked") 
public static final List<Class<? extends Pet>> allTypes = 
Collections .unmodifiableList (Arrays. asList( 
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Pet.class, Dog.class, Cat.class, Rodent.class, 
Mutt.class, Pug.class, EgyptianMau.class, Manx.class, 
Cymric.class, Rat.class, Mouse.class,Hamster.class),); 
// Types for random creation: 
private static final List<Class<? extends Pet>> types = 
all Types. subList (all Types. indexOf (Mutt.class), 
allTypes.size()): 
public List<Class<? extends Pet>> types() { 
return types; 
} 
public static void main(String[] args) { 
System. out.printIn(types) ; x 


} 
} /* Output: 
[class typeinfo.pets.Mutt, class typeinfo.pets.Pug, class 
typeinfo.pets.€gyptianMau, class typeinfo.pets.Manx, class 
typeinfo.pets.Cymric, class typeinfo.pets.Rat, class 
typeinfo.pets.Mouse, class typeinfo. pets.Hamster] 
Wm 


在 即将 出 现 的 PetCount3.java 示 例 中 ， 我 们 需要 先 用 所 有 的 Pet 类 型 来 预 加 载 一 个 Map (而 
仅仅 只 是 那些 将 要 随机 生成 的 类 型 )， 因 此 allTypes List 是 必需 的 。types 列 表 是 allTypes 的 一 部 分 
(通过 使 用 ListsubList0 创 建 的 )， 它 包含 了 确切 的 宠物 类 型 ， 因 此 它 被 用 于 随机 Pet 生 成 。 
这 一 次 ， 生 成 types 的 代码 不 需要 放 在 try 块 内 ， 因 为 它 会 在 编译 时 得 到 检查 ， 因 此 ， 它 不 会 
抛 出 任何 异常 这 与 Class:forName0 不 一 样 。 
我 们 现在 在 typeinfo.pets 类 库 中 有 了 两 种 PetCreator 的 实现 。 为 了 将 第 二 种 实现 作为 默认 实 
[57] W, 我 们 可 以 创建 一 个 使 用 了 LiteralPetCreator 的 外 观 : 


' ， 





//; typeinfo/pets/Pets. java 
// Facade to produce a default PetCreator. 
package typeinfo.pets; 

import java.util 








public class Pets { 
public static final PetCreator creator = a 
new LiteralPetCreator(); s 
public static Pet randomPet() { 
return creator.randomPet(); 


} 
public static Pet[] createArray(int size) { 
return creator.createArray (size) 





} 
public static ArrayList<Pet> arrayList(int size) { 
return creator.arrayList(size); 


} 
} Mi~ 
这 个 类 还 提供 了 对 randomPet0、createArray0 和 arrayList0 的 间接 调用 。 


因为 PetCount.countPets0 接 受 的 是 一 个 PetCreator 参 数 ， 我 们 可 以 很 容易 地 测试 LiteralPet- 
Creator (通过 上 面 的 外 观 ): 


//: typeinfo/PetCount2. java 
import typeinfo.pets.*: ` 


public class PetCount2 { 
public static void main(String[] args) { 
PetCount.countPets(Pets.creator); 
} 2 
} /* (Execute to see output) *///:~ 


该 示例 的 输出 与 PetCountjava 相 同 。 
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14.3.2 动态 的 instanceof 
Class.isInstance 方 法 提供 了 一 种 动态 地 测试 对 象 的 途径 。 于 是 所 有 那些 单调 的 instanceof 语 
名 都 可 以 从 PetCountjava 的 例子 中 移 除了 。 如 下 所 示 : 


1/: typeinfo/PetCount3. java 
// Using isInstance() 

‘import typeinfo.pets.*; 

import java.util.*; i 

import net.mindview.util.*; 

import static net.mindview.util.Print.*; 


public class PetCount3 { 
static class PetCounter 
extends LinkedHashMap<Class<? extends Pet>,Integer> { 
public PetCounter() { 
super (MapData.map(LiteralPetCreator.allTypes, ®)); 


$ 
public void count(Pet pet) { ‘ 
// Class.isInstance() eliminates instanceofs: 
for (Map.Entry<Class<? extends Pet>,Integer> pair 
: entrySet()) 
if (pair.getKey().isInstance(pet)) 
put(pair.getkey(), pair.getValue() + 1); 


} 
public String toString) { 
StringBuilder result = new StringBuilder("("); 
for (Map.Entry<Class<? extends Pet>,Integer> pair 
: entrySet()) { 
result. append (pair.getKey() .getSimpleName()) ; 
result. append("="); 
result. append (pair .getValue()).; 
result.append(", "); 


} 
result.delete(result. length()-2, result.length()); 
result.append("}"); 
return result.toString(); 
} eu 


} 
public static void main(String[] args) ( 
PetCounter petCount = new PetCounter(); 
for(Pet pet : Pets.createArray(2@)) { 
printnb(pet.getClass().getSimpleName() + " "); 
petCount..count (pet) ; 
$ 
print(); 
print (petCount) ; 
} 
} /* Output: 
Rat Manx Cymric Mutt Pug Cymric Pug Manx Cymric Rat 
EgyptianMau Hamster EgyptianMau Mutt Hutt Cymric Mouse Pug 
Mouse Cymric 
{Pet=20, Dog=6, Cat=9, Rodent=5, Mutt=3, Pug=3, _ 
EgyptianMau=2, Manx=7, Cymric=5, Rat=2, Mouse=2, Hamster=1} 
“H~ 


为 了 对 所 有 不 同类 型 的 Pet 进行 计数 ，PetCounter Map 预 加 载 了 LiteralPetCreatorallTypes 
中 的 类 型 。 这 使 用 了 net.mindview.util.MapData 类 ， 这 个 类 接受 一 个 Iteralbe (allTypes List) 和 
一 个 常数 值 ( 在 本 例 中 是 0) ， 然 后 用 allTypes 中 元 素 作为 键 ， 用 0 作为 值 ， 来 填充 Map。 如 果 不 
预 加 载 这 个 Map， 那 么 你 最 终 将 只 能 对 随机 生成 的 类 型 进行 计数 ， 而 不 包括 诸如 Pet 和 Cat 这 样 
的 基 类 型 。 

可 以 看 到 ，isInstance0 方 法 使 我 们 不 再 需 朗 instanceof 表 达 式 。 此 外 ， 这 意味 着 如 果 要 求 添 
加 新 类 型 的 Pet， 只 需 简单 地 改变 LiteralPetCreatorjava 数 组 即 可 ， 而 毋 需 改 动 程序 其 他 部 分 
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(但 是 在 使 用 istanceof 时 这 是 不 可 能 的 ) 。 
toString0 方 法 已 经 被 重 载 ， 使 得 输出 更 容易 被 读 取 ， 而 该 输出 与 打印 Map 时 所 看 到 的 典型 
输出 仍然 是 匹配 的 。 


14.3.3 递归 计数 
在 PetCount3.PetCounter 中 的 Map 预 加 载 了 所 有 不 同 的 Pet 类 。 与 预 加 载 映射 表 不 同 的 是 ， 
我 们 可 以 使 用 ClassisAssignableFrom0， 并 创建 一 个 不 局 限于 对 Pet 计数 的 通用 工具 。 


11:_net/mindview/utit/TypeCounter.java 
// Counts instances of a type family. 

package net.mindview.util: 7 
import java.util.*; 


public class TypeCounter extends HashMap<Class<?>, Integer>{ 
private Class<?> baseType; 
public TypeCounter (Class<?> baseType) { 
this.baseType = baseType; 


} 
public void count (Object obj) { 
Class<?> type = obj .getClass(); 
if (IbaseType. isAssignableFrom(type)) 
throw new RuntimeException(obj +" incorrect type: " 
+ type +", should be type or subtype of * 
+ baseType); 
countClass(type) : 


} 
private void countClass(Class<?> type) { 
Integer quantity = get (type); 
put(type, quantity == null ? 1 : quantity + 1); 
Class<?> superClass = type.getSuperclass(); 
if(superClass != null && 
baseType. isAssignableFrom(superClass) ) 
countClass (superClass) ; 








} 
public String toString() { 
StringBuilder result = new StringBuilder("{"): 
for (Map. Entry<Class<?>,Integer> pair : entrySet()) { 
result. append(pair.getKey() .getSimpleNamé()); 
result. append ("="); 
result. append(pair.getValue()); 
result.append(", "); 


li 

result.delete(result.length()-2, result. length()); 
result.append("}"); 

return result. toString(); 


} 
} ti~ 


count0 方 法 获取 其 参数 的 Class， 然 后 使 用 isAssignableFrom0 来 执行 运行 时 的 检查 ， 以 校 验 
你 传递 的 对 象 确实 属于 我 们 感 兴趣 的 继承 结构 。countClass0 首 先 对 该 类 的 确切 类 型 计数 ， 然 后 ， 
如 果 其 超 类 可 以 赋值 给 baseType，countClass0 将 其 超 类 上 递归 计数 。 ae 


//: typeinfo/PetCount4. java 
import, typeinfo.pets.* 
import net.mindview.util.*; 
import static net.mindview.util.Print. 








public class PetCount4 { 
public static void main(String(] args) { 
TypeCounter counter = new TypeCounter(Pet.class); 
for (Pet pet : Pets.createArray(20)) { 
printnb(pet.getClass().getSimpleName() + " "): 
counter .count (pet) ; . 


} 
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print(); 
print (counter) ; 


} 
} /* Output: (Sample) 
Rat Manx Cymric Mutt Pug Cymric Pug Manx Cymric Rat 
Egyptiantau Hamster EgyptianMau Mutt Mutt Cymric Mouse Pug 
Mouse Cymric 
{Mouse=2, Dog=6, Manx=7, EgyptianMau=2, Rodent=5, Pug=3, 
Mutt=3, Cymric=5, Cat=9, Hamster=1, Pet=20, Rat=2) 
Ii~ 


正如 在 输出 中 看 到 的 那样 ， 对 基 类 型 和 确切 类 型 都 进行 了 计数 。 

练习 11， (2) 在 typeinfo.pets 类 库 中 添加 Gerbil， 并 修改 本 章 中 的 所 有 示例 ， 让 它们 适应 这 个 
新 类 。 

练习 12，(3) 将 第 15 章 中 的 CoffeeGeneratorjava 类 用 于 TypeCounter。 

练习 13，(3) 将 本 章 中 的 RegisteredFactoriesjava 示 例 用 于 TypeCounter。 


14.4 注册 工厂 ed 


生成 Pet 继承 结构 中 的 对 象 存在 着 一 个 问题 ， 即 每 次 向 该 继承 结构 添加 新 的 Pet 类 型 时 ， 必 
须 将 其 添加 为 LiteralPetCreatorjava 中 的 项 。 如 果 在 系统 中 已 经 存在 了 继承 结构 的 常规 的 基础 ， 
然后 在 其 上 要 添加 更 多 的 类 ， 那 么 就 有 可 能 会 出 现 问题 。 

你 可 能 会 考虑 在 每 个 子 类 中 添加 静态 初始 化 器 ， 以 使 得 该 初始 化 器 可 以 将 它 的 类 添加 到 某 
个 List 中 。 遗 憾 的 是 ， 静 态 初始 化 器 只 有 在 类 首先 被 加 载 的 情况 下 才能 被 调用 ， 因 此 你 就 磁 上 了 
“ 先 有 鸡 还 是 先 有 蛋 ” 的 问题 : 生成 器 在 其 列表 中 不 包含 这 个 类 ， 因 此 它 永 远 不 能 创建 这 个 类 的 
对 象 ， 而 这 个 类 也 就 不 能 被 加 载 并 置 于 这 个 列表 中 。 

这 主要 是 因为 ， 你 被 强制 要 求 自己 去 手工 创建 这 个 列表 (除非 你 想 编写 一 个 工具 ， 它 可 以 
全 面 搜索 和 分 析 源 代码 ， 然 后 创建 和 编译 这 个 列表 )。 ， 你 最 佳 的 做 法 是 ， 将 这 个 列表 置 于 
一 个 位 于 中 心 的 、 位 置 明显 的 地 方 ， 而 我 们 感 兴趣 的 继承 结构 的 基 类 可 能 就 是 这 个 最 佳 位 置 。 

这 里 我 们 需要 做 的 其 他 修改 就 是 使 用 工厂 方法 设计 模式 ， 将 对 象 的 创建 工作 交 给 类 自己 去 
完成 。 工 厂 方法 可 以 被 多 态 地 调用 ， 从 而 为 你 创建 恰当 类 型 的 对 象 。 在 下 面 这 个 非常 简单 的 版 
本 中 ， 工 厂 方法 就 是 Factory 接 口中 的 create( 方 法 : 

1/: typeinfo/factory/Factory. java 


package typeinfo. factory; 
public interface Factory<T> { T create(); ) ///:~ 


泛 型 参数 T 使 得 ereate0 可 以 在 每 种 Factory 实 现 中 返回 不 同 的 类 型 。 这 也 充分 利用 了 协 变 返 
回 类 型 。 

在 下 面 的 示例 中 ， 基 类 Part 包 含 一 个 工厂 对 象 的 列表 。 对 于 应 这 个 由 createRandom() 方 法 
产生 的 类 型 ， 它 们 的 工厂 都 被 添加 到 了 partFactories List 中 ， 从 而 被 注册 到 了 基 类 中 ， 


//: typeinfo/RegisteredFactories.java 
// Registering Class Factories in the base class. 





import typeinfo. factory.*; 
import java.util.*; 


class Part { 
public String toString() { 二 

return getClass().getSimpleName() ; 

} 

static List<Factory<? extends Part>> partFactories = 
new ArrayList<Factory<? extends Part>>(): 

static { 
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// Collections.addAll() gives an “unchecked generic 
// array creation ... for varargs parameter" warning. 
partFactories.add(new FuelFilter.Factory()); 
partFactories.add(new AirFilter.Factory()); 
partFactories.add(new CabinAirFilter.Factory()): 
partFactories.add(new OilFilter.Factory()): 
partFactories.add(new FanBelt.Factory()); 
partFactories.add(new PowerSteer ingBelt.Factory()); 
partFactories.add(new GeneratorBelt .Factory()); 

} 

private static Random rand = new Random(47); 

public static Part createRandom() { 
int n = rand.nextInt(partFactories.size()); 

583 return partFactories.get(n).create(); 


} 
} . 


class Filter extends Part {} 





class FuelFilter extends Filter { 
// Create a Class Factory for each specific type: 
public static class Factory 
implements typeinfo.factory.Factory<FuelFilter> { 
public FuetFilter create() { return new FuelFilter(); } 


1 
} 
class AirFilter extends Filter { 
' public static class Factory 
k implements typeinfo. factory.Factory<AirFilter> { 
public AirFilter create() { return new AirFilter(); } 
} s 


} 


class CabinAirFilter extends Filter { 
public static class Factory 
implements typeinfo.factory ,FactorycCabinAfrFitter> ( 
public CabinAirFilter create() { 
return new CabinAirFilter() ; 
> 
} 
} 


class OilFilter extends Filter { 
public static class Factory 
implements typeinfo. factory.Factory<OilFilter> { 
public OilFilter create() { return new OilFilter(); } 
} 
} 


class Belt extends Part {} 


class FanBelt extends Belt { 
public static class Factory 
implements typeinfo.factory.Factory<FanBelt> { 
public FanBelt create() { return new FanBelt(): } 
} 


fal} 


class GeneratorBelt extends Belt { 
public static class Factory 
implements typeinfo. factory.Factory<GeneratorBelt> { 
public GeneratorBelt create() {7 
return new GeneratorBelt(); 
| > 
1 
} 
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class PowerSteeringBelt extends Belt { ¥ i eso x A 
public static class Factory Se x 
implements typeinfo.tactory.Factory<PowerSteeringBelt> { art et 

public PowerSteeringBelt create() { D a3 
return new PowerSteeringBelt(); a. a a 
} XY tty ‘ i 
} i Eik et 4 
) ‘ 


public class RegisteredFactories { 
public static void main(String[] args) { 
for(int 1 = 0; 1 < 10; i++) 2 
System.out.print1n(Part.createRandom()); 


} s EET 
} /* Output: i sat 
GeneratorBelt ， + 
CabinAirFilter 
GeneratorBelt . 
AirFilter one r 
PowerSteeringBelt ` 
CabinAirFilter th 4 
FuelFilter 人 È 
PowerSteeringBelt . x - 
Power Steer ingBelt EvS 
FuelFilter sea sts 
Whim 


并 非 所 有 在 继承 结构 中 的 类 都 应 该 被 实例 化 ， 在 本 例 中 ， Filteri RR, 因此 
你 不 应 该 创建 它们 的 实例 ， 而 只 应 该 创建 它们 的 子 类 的 实例 。 如 果 某 个 类 应 该 由 createRandom() 
方法 创建 ， 那 么 它 就 包含 一 个 内 部 Factory 类 。 如 上 所 示 ， 重用 名 字 Factory 的 唯一 方式 就 是 限定 
typeinfo.factory.Factory, 

尽管 你 可 以 使 用 Collections.addAll0 来 向 列表 中 添加 工厂 ， 但 是 这 样 做 编译 器 就 会 表达 它 的 
不 满 ， 抛 出 一 条 有 关 “ 创 建 泛 型 数组 ”( 这 被 认为 是 不 可 能 的 ， 正 如 你 将 在 第 15 章 中 所 看 到 的 那 
样 ) 的 警告 ， 因 此 我 转 而 使 用 add0。createRandom() 方 法 从 partFactories 中 随机 地 选取 一 个 工 
广 对 象 ， 然 后 调用 其 create0 方 法 ， 从 而 产生 一 个 新 的 Part。 ， 

练习 14，(4) 构造 器 就 是 一 种 工厂 方法 。 人 egeteredF ectetice avn, 全 其 不 要 使 用 显 式 
的 工厂 ， 而 是 将 类 对 象 存储 到 List 中 ， 并 使 用 newInstance0 来 创建 对 象 。 

练习 15: (4) 使 用 注册 工厂 来 实现 一 个 新 的 PetCreator， 并 修改 Pets 外 观 ， 使 其 使 用 这 个 新 
的 Creator 而 并 非 另外 两 个 Creator。 确 保 使 用 Petsjava 的 其 他 示例 仍 可 以 正常 工作 。 : 

练习 16: (4) 修改 第 15 章 中 的 Coffee 继 承 结构 ， 以 便 可 以 使 用 注册 工厂 。 ` 


14.5 instanceof 与 Class 的 等 价 性 > 


在 查询 类 型 信息 时 ， 以 instanceof 的 形式 〈 即 以 instanceof 的 形式 或 isInstanee0) 的 形式 ， 它 们 
产生 相同 的 结果 ) 与 直接 比较 Class 对 象 有 一 个 很 重要 的 差别 。 TA Aad 


+ 1/: typeinfo/Fami lyWstxactType. java 
1/ The difference between instanceof and class 
~ package typeinfo: 
import static net.mindview.util.Print.*; “上 D es E x 


$ 9 


class Base {} Sat 
class Derived extends Base {} : w x 


1 Public class FamilyVsExactType { A a he! oe 


static void test(Qbject x) { 
print("Testing x of type "e+ x.getClass()); 
print("x instanceof Base " + (x instanceof Base); ~~ ` $ 

















|586] 








加 
3 


8 
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print("x instanceof Derived "+ (x instanceof Derived)); + 
print("Base.isInstance(x) "+ Base.class.isInstance(x)); 
print (Derived. isinstance(x) “+ 

Derived.class. isInstance(x)); 
print(*x.getClas: jase.class " + 
e.class)); 






(x.getClass() == Derived.class)); 
print("x.getClass().equals(Base.class)) “+ 
(x.getClass() .equals(Base. class))); 
print("x.getClass() .equals(Derived.class)) "+ 
(x, getClass () .equals (Derived.class))): 


} 

public static void main(String[] args) { 
test (new Base()); 
test (new Derived()); 


} 
} /* Output: 
Testing x of type class typeinfo.Base 
x instanceof Base true 
x instanceof Derived false 
Base. isInstance(x) true 
Derived. isinstance(x) false 
x.getClass() == Base.class true 
x.getClass() == Derived.class false 
x.getClass() .equals(Base.class)) true 
x.getClass() .equals(Derived.class)) false 
Testing x of type class typeinfo. Derived 
x instanceof Base true 
x instanceof Derived true 
Base. isInstance(x) true 
Derived. isInstance(x) true 
x.getClass() == Base.class false 
x.getClass() == Derived.class true 
x.getClass() equals (Base.class)) false 
x. getClass () equals (Derived.class)) true 
Wha 


test0 方 法 使 用 了 两 种 形式 的 instanceof 作 为 参数 来 执行 类 型 检查 。 然 后 获取 Class 引 用 ， 并 
用 == 和 equals0 来 检查 Class 对 象 是 否 相等 。 使 人 放心 的 是 ，instancof 和 isInstance0 生 成 的 结果 完 
全 一 样 ，equals0 和 == 也 一 样 。 但 是 这 两 组 测试 得 出 的 结论 却 不 相同 。instanceof 保 持 了 类 型 的 
概念 ， 它 指 的 是 “你 是 这 个 类 吗 ， 或 者 你 是 这 个 类 的 派生 类 吗 ? ”而 如 果 用 == 比 较 实际 的 Class 
对 象 ， 就 没有 考虑 继承 ~ 一 它 或 者 是 这 个 确切 的 类 型 ， 或 者 不 是 。 


14.6 Rat: 运行 时 的 类 信息 


如 果 不 知道 某 个 对 象 的 确切 类 型 ，RTTI 可 以 告诉 你 。 但 是 有 一 个 限制 ， 这 个 类 型 在 编译 时 
必须 已 知 ， 这 样 才能 使 用 RTTI 识 别 它 ， 并 利用 这 些 信息 做 一 些 有 用 的 事 。 换 句 话说 ， 在 编译 时 ， 
编译 器 必须 知道 所 有 要 通过 RTTI 来 处 理 的 类 。 

初 看 起 来 这 似乎 不 是 个 限制 ， 但 是 假设 你 获取 了 一 个 指向 某 个 并 不 在 你 的 程序 空间 中 的 对 
象 的 引用 ， 事 实 上 ， 在 编译 时 你 的 程序 根本 没 法 获知 这 个 对 象 所 属 的 类 。 例 如 ， 假 设 你 从 磁盘 
文件 ， 或 者 网 络 连接 中 获取 了 一 串 字 节 ， 并 且 你 被 告知 这 些 字 节 代表 了 一 个 类 。 既 然 这 个 类 在 
编译 器 为 你 的 程序 生成 代码 之 后 很 久 才 会 出 现 ， 那 么 怎样 才能 使 用 这 样 的 类 呢 ? 

在 传统 的 编程 环境 中 不 太 可 能 出 现 这 种 情况 。 但 当 我 们 置身 于 复 大 规模 的 编程 世界 中 ， 在 
许多 重要 情况 下 就 会 发 生 上 面 的 事情 。 首 先 就 是 “基于 构件 的 编程 "， 在 此 种 编程 方式 中 ， 将 使 
用 某 种 基于 快速 应 用 开发 (RAD) 的 应 用 构建 工具 ， 即 集成 开发 环境 (IDE)， 来 构建 项 目 。 这 
是 一 种 可 视 化 编程 方法 ， 可 以 通过 将 代表 不 同 组 件 的 图 标 拖 点 到 表单 中 来 创建 程序 。 然 后 在 编 
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程 时 通过 设置 构件 的 属性 值 来 配置 它们 。 这 种 设计 时 的 配置 ， 要 求 构件 都 是 可 实例 化 的 ， 并 且 
要 暴露 其 部 分 信息 ， 以 人 允许 程序 员 读 取 和 修改 构件 的 属性 。 此 外 ， 处 理 图 形 化 用 户 界面 (GUI) 
事件 的 构件 还 必须 暴露 相关 方法 的 信息 ， 以 便 IDE 能 够 帮助 程序 员 覆 盖 这 些 处 理事 件 的 方法 。 反 
射 提供 了 一 种 机 制 一 一 用 来 检查 可 用 的 方法 ， 并 返回 方法 名 。Java 通 过 JavaBeans (第 22 章 将 详 
细 介 绍 ) 提供 了 基于 构件 的 编程 架构 。 

人 们 想 要 在 运行 时 获取 类 的 信息 的 另 一 个 动机 ， 便 是 希望 提供 在 路 网 络 的 远程 平台 上 创建 
和 运行 对 象 的 能 力 。 这 被 称 为 远程 方法 调用 (RMI) ， 它 允许 一 个 Java 程 序 将 对 象 分 布 到 多 台 机 
器 上 。 需 要 这 种 分 布 能 力 是 有 许多 原因 的 ， 例 如 ， 你 可 能 正在 执行 一 项 需 进行 大 量 计算 的 任务 ， 
为 了 提高 运算 速度 ， 想 将 计算 划分 为 许多 小 的 计算 单元 ， 分 布 到 空闲 的 机 器 上 运行 。 又 比如 ， 
你 可 能 希望 将 处 理 特定 类 型 任务 的 代码 (例如 多 层 的 C/S (客户 /服务 器 ) 架构 中 的 “业务 规则 ”)， 
置 于 特定 的 机 器 上 ， 于 是 这 台 机 器 就 成 为 了 描述 这 些 动作 的 公共 场所 ， 可 以 很 容易 地 通过 改动 
它 就 达到 影响 系统 中 所 有 人 的 效果 。( 这 是 一 种 有 趣 的 开发 方式 ， 因 为 机 器 的 存在 仅仅 是 为 了 方 
便 软件 的 改动 ! ) 同时 ， 分 布 式 计算 也 支持 适 于 执行 特殊 任务 的 专用 硬件 ， 例如 矩阵 转 置 ， 而 
这 对 于 通用 程序 就 显得 不 太 合 适 或 者 太 昂贵 了 。 

Class 类 与 java.lang.reflect 类 库 一 起 对 反射 的 概念 进行 了 支持 ， 该 类 库 包含 了 Field 、 
Method 以 及 Constructor 类 (每 个 类 都 实现 了 Member 接 口 )。 这 些 类 型 的 对 象 是 由 JVM 在 运行 
时 创建 的 ， 用 以 表示 未 知 类 里 对 应 的 成 员 。 这 样 你 就 可 以 使 用 Constructor 创 建新 的 对 象 ， 用 
get() 和 set0 方 法 读 取 和 修改 与 Field 对 象 关联 的 字段 ， 用 invoke0 方 法 调用 与 Method 对 象 关联 的 
方法 。 另 外 ， 还 可 以 调用 getFields0、getMethods0 和 getConstructors0) 等 很 便利 的 方法 ， 以 返 
回 表示 字段 、 方 法 以 及 构造 器 的 对 象 的 数组 (在 JDK 文 档 中 ， 通 过 查找 Class 类 可 了 解 更 多 相关 

料 )。 这 样 ， 匿 名 对 象 的 类 信息 就 能 在 运行 时 被 完全 确定 下 来 ， 而 在 编译 时 不 需要 知道 任何 
事情 。 

重要 的 是 ， 要 认识 到 反射 机 制 并 没有 什么 神奇 之 处 。 当 通过 反射 与 一 个 未 知 类 型 的 对 象 打 
交道 时 ，JVM 只 是 简单 地 检查 这 个 对 象 ， 看 它 属于 哪个 特定 的 类 (就 像 RTTI 那 样 )。 在 用 它 做 其 
他 事情 之 前 必须 先 加 载 那个 类 的 Class 对 象 ， 那 个 类 的 .class 文 件 对 于 JVM 来 说 必须 是 可 获 
取 的 : 要 么 在 本 地 机 器 上 ， 要 么 可 以 通过 网 络 取得 。 所 以 RTTI 和 反射 之 间 真 正 的 区 别 只 在 于 ， 
对 RTTI 来 说 ， 编 译 器 在 编译 时 打开 和 检查 .class 文 件 。( 换 句 话说 ,我 们 可 以 用 “普通 ”方式 调 
用 对 象 的 所 有 方法 。) 而 对 于 反射 机 制 来 说 ，.class 文 件 在 编译 时 是 不 可 获取 的 ， 所 以 是 在 运行 时 
打开 和 检查 .class 文 件 。 

14.6.1 类 方法 提取 器 

通常 你 不 需要 直接 使 用 反射 工具 ， 但 是 它们 在 你 需要 创建 更 加 动态 的 代码 时 会 很 有 用 。 反 
射 在 Java 中 是 用 来 支持 其 他 特性 的 , 例如 对 象 序列 化 和 JavaBean (它们 在 本 书 稍 后 部 分 都 会 提 到 )。 
但 是 ， 如 果 能 动态 地 提取 某 个 类 的 信息 有 的 时 候 还 是 很 有 用 的 。 请 考虑 类 方法 提取 器 。 浏 览 实 
现 了 类 定义 的 源 代码 或 是 其 JDK 文 档 ， 只 能 找到 在 这 个 类 定义 中 被 定义 或 被 覆盖 的 方法 。 但 对 你 
来 说 ， 可 能 有 数 十 个 更 有 用 的 方法 都 是 继承 自 基 类 的 。 要 找 出 这 些 方法 可 能 会 很 乏味 且 费 时 9 。 
幸运 的 是 ， 反 射 机 制 提供 了 一 种 方法 ， 使 我 们 能 够 编写 可 以 自动 展示 完整 接口 的 简单 工具 。 下 
面 就 是 其 工作 方式 : 


//: typeinfo/ShowĦethods. java 
// Using reflection to show all the methods of a class, 
// even if the methods are defined in the base class. 





O 特别 是 在 过 去 ， 现 在 Sun 已 极 大 地 改进 了 其 HTML Java 文档 ， 所 以 查找 基 类 的 方法 已 经 简单 多 了 。 


[588| 
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I> 7/ {Args: ShowMethods} 4 
import java.lang.reflect.*; P 
import. java.util.regex.*; x 
import static fet..mindview.util.Print. 








， pubtie class-ShowMethods {  ， 
private static String usage = 
“usage: \n" + 
"ShowMethods qualified.class.name\n" + 
To show all methods in class or:\n" + 
"ShowMethods qualified.class.ndme Word\n" + . 
"To search for methods involving ‘word'*; 
. private static Pattern p = Pattern.compile(*\\w+\\. 
Public static void main(String[] args) { 
if(args.tength < 1) ¢ 
print(usage); 
System.exit (8); 
} 
int lines = 0; 
(590) sya 
a Class<?> < = Class. forName(args{0]); + 
Method[] methods = c.getMethods(); ` 
Constructor[] ctors = c.getConstructors(); 
t if (args.length == 1) ( 
for (Method method : methods) 
print( 
p.matcher (method. toString()).replaceAll("")); 
for (Constructor ctor : ctors) 
print(p.matcher (ctor. toString()) .replaceAll("")); 
Vnes = methods. length + ctors.length; 
} else { 
for (Method method : methods) 
if (method. toString().indexof(args(1}) != -1) { 
Š print( . 
p.matcher (method. toString()) .replaceAtl(**)): 
Lines++; 
} 
for(Constructor ctor : ctors) 
if(ctor.toString() .indexof(args[1]) != -1) { 
brint(p.matcher¢ 
ctor. toString()).replacealt(*")); 
lines++; 











a 2 
} catch(ClassNotFoundException e) { 
A print("Nò such class! ”+ e); 
} 
} 
} /* Output: 
public static void main(String{)) 
public native int hashCode¢) 
public final native Class getClass() 
public final void wait(long, int) throws 
InterruptedException i 
public final void wait() throws InterruptedException 
public final native void wait(long) throws 
Interruptedexcept ion 
public boolean equals (Object) 
public String toString() 
public final native void notify() 
public final native void notifyAll() 
public ShowMethods() 
[591 Whew 2 











Class 的 getMethods0 和 getConstractors0) 方 法 分 别 返 回 Method 对 象 的 数组 和 Constructor 
对 象 的 数组 。 这 两 个 类 都 提供 了 深层 方法 ， 用 以 解析 其 对 象 所 代表 的 方法 ， 并 获取 其 名 字 
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人 参数 以 及 返回 值 。 但 也 可 以 像 这 里 一 样 ， 只 使 用 toStringO 生 成 一 个 含有 完整 的 方法 特征 签 
名 的 字符 种。 代码 其 他 部 分 用 于 提取 命令 行 信息 ， 判 断 某 个 特定 的 特征 签名 是 否 与 我 们 的 目标 
字符 串 相符 〈 使 用 indexOfO) ， 并 使 用 正则 表达 式 去 掉 了 命名 修饰 词 正则 表达 式 在 第 13 章 中 
介绍 过 )。 

Class.forName0 生 成 的 结果 在 编译 时 是 不 可 知 的 ， 因 此 所 有 的 方法 特征 签名 信息 都 是 在 执 
行 时 被 提取 出 来 的 。 如 果 研 究 一 下 JDK 文 档 中 关于 反射 的 部 分 ， 就 会 看 到 ， 反 射 机 制 提供 了 足 
够 的 支持 ， 使 得 能 够 创建 一 个 在 编译 时 完全 未 知 的 对 象 ， 并 调用 此 对 象 的 方法 (在 本 书后 面 会 
有 示例 ) 。 虽 然 开始 的 时 候 可 能 认为 永远 也 不 需要 用 到 这 些 功能 ， 但 是 反射 机 制 的 价值 是 很 惊 
人 的 。 

上 面 的 输出 是 从 下 面 的 命令 行 产 生 的 : a v 

java ShowMethods ShowMethods ud j 
你 可 以 看 到 ， 输 出 中 包含 一 个 publie 的 默认 构造 器 ， 即便 能 在 代码 中 看 到 根本 没有 定义 任何 构造 
器 。 所 看 到 的 这 个 包含 在 列表 中 的 构造 器 是 编译 器 自动 合成 的 。 如 果 将 ShowMethods 作 为 一 个 
非 public 的 类 (也 就 是 拥有 包 访问 权限 )， 输 出 中 就 不 会 再 显示 出 这 个 自动 合成 的 默认 构造 器 了 。 
该 自动 合成 的 默认 构造 器 会 自动 被 赋予 与 类 一 样 的 访问 权限 。 

还 有 一 个 有 趣 的 例子 是 ， 用 一 个 额外 的 char、 int 或 String 等 参数 来 调用 java ShowMethods 
java.lang.String, 

在 编程 时 ， 特 别 是 如 果 不 记 得 一 个 类 是 否 有 某 个 方法 ， 或 者 不 知道 一 个 类 究竟 能 做 些 什么 
例如 Color 对 象 ， 而 又 不 想 通过 索引 或 类 的 层次 结构 去 查找 JDK 文 档 ， 这 时 这 个 工具 确实 能 节省 
很 多 时 间 。 ‘ 

第 22 章 包含 这 个 程序 的 GUI 版 本 ( 专 为 提取 Swing 构件 的 信息 而 定制 的 ) ， 使 你 在 编写 代码 
的 同时 能 够 通过 运行 它 来 快速 查询 有 用 的 信息 。 

练习 17，(2) 修改 ShowMethodsjava 中 的 正则 表达 式 ， 以 去 掉 native 和 final 关 键 字 (提示: 
使 用 “或 ”运算 符 “『) 。 

练习 18: (1) 将 ShowMethods 变 为 一 个 非 public 的 类 ， 并 验证 合成 的 默认 构造 器 不 会 再 在 答 
出 中 出 现 。 

练习 19，(4) 在 ToyTestjava 中 ， 使 用 反射 机 制 ， 通 过 非 默认 构造 器 创建 Toy 对 象 。 

练习 20，(5) 请 从 在 http://java.sun.com 上 提供 的 JDK 文 档 中 找 出 java.lang.Class 的 接口 。 写 一 
个 程序 ， 使 它 能 够 接受 命令 行 参 数 所 指定 的 类 名 称 。 然 后 使 用 Class 的 方法 打印 该 类 所 有 可 以 获 
得 的 信息 。 用 标准 程序 库 的 类 和 你 自己 写 的 类 ， 分 别 测试 这 个 程序 。 


14.7 动态 代理 


代理 是 基本 的 设计 模式 之 一 ， 它 是 你 为 了 提供 额外 的 或 不 同 的 操作 ， 而 插 人 的 用 来 代替 
“实际 ”对 象 的 对 象 。 这 些 操作 通常 涉及 与 “实际 ”对 象 的 通信 ， 因 此 代理 通常 充当 着 中 间 人 的 
角色 。 下 面 是 一 个 用 来 展示 代理 结构 的 简单 示例 : * 


//: typeinfo/SimpleProxyDemo. java 
import static net.mindview.util.Print.*; 


interface Interface { 

void doSomething(); 

void somethingElse(String arg); 
d 


class RealObject implements Interface { 
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public void doSomething() { print("doSomething’); 


public void somethingElse(String arg) { 
print("somethingElse ”+ arg); 
roi 
t} 
class SimpleProxy implements Interface { 
private Interface proxied; 


public SimpleProxy(Interface proxied) { 
` this. proxied = proxied: 


z H 


public void doSomething() { . 
print ("SimpleProxy doSomething"); 
proxied.doSomething() ; 


} 

Public void somethingElse(String arg) { 
print("SimpleProxy somethingElse ”+ arg); 
proxied, somethingElse(arg); 

六 


"class SimpleProxyDemo. { 
public static void consumer(Interface iface) { 
iface.doSomething(); 
iface. somethingE1se("bonobo") ; 


public static void main(String[] args) { 
consumer (new RealObject()); 
consumer (new SimpleProxy(new RealObject())); 


} 

/* Output: 

doSomething 

somethingElse bonobo 

SimpleProxy doSomethihg 
doSomething 

SimpleProxy somethingElse bonobo 


somethingElse bonobo 
Whim 


因为 consumer() 接 受 的 Interface， 所 以 它 无 法 知道 正在 获得 的 到 底 是 RealObject 还 是 
SimpleProxy， 因 为 这 二 者 都 实现 了 Interface。 但 是 SimpleProxy 已 经 被 插入 到 了 客户 端 和 
RealObject 之 间 ， 因 此 它 会 执行 操作 ， 然 后 调用 RealObject 上 相同 的 方法 。 

在 任何 时 刻 ， 只 要 你 想 要 将 额外 的 操作 从 “实际 ”对 象 中 分 离 到 不 同 的 地 方 ， 特 别 是 当 你 
希望 能 够 很 容易 地 做 出 修改 ， 从 没有 使 用 额外 操作 转 为 使 用 这 些 操作 ， 或 者 反 过 来 时 ， 代 理 就 
显得 很 有 用 (设计 模式 的 关键 就 是 封装 修改 一 -因此 你 需要 修改 事务 以 证 明 这 种 模式 的 正确 性 )。 
例如 ， 如 果 你 希望 跟踪 对 RealObject 中 的 方法 的 调用 ， 或 者 希望 度量 这 些 调用 的 开销 ， 那 么 你 
应 该 怎样 做 昵 ? 这些 代 码 肯定 是 你 不 希望 将 其 合并 到 应 用 中 的 代码 ， 因 此 代理 使 得 你 可 以 很 容 


易 地 添加 或 移 除 它们 。 


Java 的 动态 代理 比 代理 的 思想 更 向 前 迈进 了 一 步 ， 因 为 它 可 以 动态 地 创建 代理 并 动态 地 处 理 
对 所 代理 方法 的 调用 。 在 动态 代理 上 所 做 的 所 有 调用 都 会 被 重 定向 到 单一 的 调用 处 理 器 上 ， 它 


} 


的 工作 是 揭示 调用 的 类 型 并 确定 相应 的 对 策 。 下 面 是 用 动态 代理 重 写 的 SimpleProxyDemojava， 


//: typeinfo/SimpleDynamicProxy.java 
import java.lang.reflect.*; 


class DynamicProxyHandler implements InvocationHandler { 


private Object proxied; 
public DynamicProxyHandler (Object proxied) { 
this.proxied = proxied; 
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} 
public Object 
invoke (Object proxy, Method method, Object{] args) 
throws Throwable { 
System.out.printin("**** proxy: ”+ proxy.getClass() + 
*, method: ”+ method + ", args: ”+ args): 
if(args != null) 
for (Object arg : args) ， ' 
System.out.printin(" ”+ arg); 
return method. invoke(proxied, args): 
} 
} ý 2 


class SimpleDynamicProxy { r 
public static void consumer (Interface iface) { 
1face.doSomething():; 
iface. somethingE1se("bonobo") ; 
} 
public static void main(String[] args) { 
RealObject real = new RealObject(); 
consumer (real); 
// Insert a proxy and call again: 
Interface proxy = (Interface) Proxy .newProxyInstance( 
Interface.class.getClassLoader(), 
new Class{]{ Interface.class }, 
new DynamicProxyHandler(real)); 
consumer (proxy) ; 


} 

} /* Output: (95% match) $ 1 

doSomething i 

somethingElse bonobo 

**** proxy: class $Proxy®, method: public abstract void 

Interface.doSomething(), args: null 

doSomething 

**** proxy: class $Proxy@, method: public abstract void 

Interface. somethingElse (java.lang.String), args: 

[Ljava. lang. Object ;@42¢816 £ 
bonobo 

somethingElse bonobo 

Whim X 


通过 调用 静态 方法 Proxy.newProxyInstanceO 可 以 创建 动态 代理 ， 这 个 方法 需要 得 到 一 个 类 
加 载 器 (你 通常 可 以 从 已 经 被 加 载 的 对 象 中 获取 其 类 加 载 器 ， 然 后 传递 给 它 ) ， 一 个 你 希望 该 代 
理 实现 的 接口 列表 (不 是 类 或 抽象 类 ) ， 以 及 InvocationHandler 接 口 的 一 个 实现 。 动 态 代理 可 以 
将 所 有 调用 重 定向 到 调用 处 理 器 ， 因 此 通常 会 向 调用 处 理 器 的 构造 器 传递 给 一 个 “实际 ”对 象 
的 引用 ， 从 而 使 得 调用 处 理 器 在 执行 其 中 介 任 务 时 ， 可 以 将 请 求 转 发 。 

invoke( 方 法 中 传递 进来 了 代理 对 象 ， 以 防 你 需要 区 分 请 求 的 来 源 ， 但 是 在 许多 情况 下 ， 你 
并 不 关心 这 一 点 。 然 而 ， 在 invoke0 内 部 ， 在 代理 上 调用 方法 时 需要 格外 当心 * 因为 对 接口 的 调 
用 将 被 重 定向 为 对 代理 的 调用 。 

通常 ， 你 会 执行 被 代理 的 操作 ， 然 后 使 用 Method.invoke0 将 请 求 转发 给 被 代理 对 象 ， 并 传 
和 人 必需 的 参数 。 这 初 看 起 来 可 能 有 些 受 限 ， 就 像 你 只 能 执行 泛 化 操作 一 样 。 但 是 ， 你 可 以 通过 
传递 其 他 的 参数 ， 来 过 滤 某 些 方法 调用 : 


//: typeinfo/SelectingMethods. java 

/4 Looking for particular methods in a dynamic proxy. 
| import java.lang.reflect.*; 

import static net.mindview.util.Print.*; 


class MethodSelector implements InvocationHandler { 
private Object proxied; 
i > public MethodSelector (Object proxied) { 
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this.proxied = proxied: 
} 


public Object om A 
invoke(Object proxy, Method method, Objectt]: args) E 
throws Throwable { i anit 


if (method. getName() .equals(“interesting” » 
print("Proxy detected the interesting method") ; 
return method. invoke(proxied, args); 
} we 


) R 1 


interface SomeMethods { 
void boringi(); 
void boring2(); 
void interesting(String arg); - x 
void boring3(); 
} . 


class Implementation implements SomeMethods { 
public void boringl() { print(“boringi"); } . : b 
public void boring2() { print("boring2"); } 
public void interesting(String arg) { ` 
print("interesting ”+ arg); 8 ~ 


} 
public void boring3() { print("boring3"); } 
} 


class SelectingMethods { 
public static void main(String() args) { a 四 
SomeNethods proxy= (SomeHethods )Proxy.newProxyInstance( 
SomeMethods .class.getClassLoader(), ‘ne 
new Class[]{ SomeMethods.class }, < 
new HethodSelector (new Implementation())); 
proxy. bor ing1(); 
proxy. boring2() ; a 
Proxy. interest ing("bonobo") ; 
proxy .boring3(); 


} 
} /* Output: 
boringt 
boring? 
Proxy detected the interesting method - 
interesting bonobo 
bor ing3 
Wh 


这 里 ， 我 们 只 查看 了 方法 名 ， 但 是 你 还 可 以 查看 方法 签名 的 其 他 方面 ， 甚 至 可 以 搜索 特定 
的 参数 值 。 
， 动态 代理 并 非 是 你 日 常 使 用 的 工具 ， 但 是 它 可 以 非常 好 地 解决 某 些 类 型 的 问题 。 你 在 
(Thinking in Patterns》( 查 看 www.MindView.net) 和 Erich Gamma 等 人 撰写 的 《Design 
Patterns) 这 两 本 书 中 了 解 到 有 关 代 理 和 其 他 设计 模式 的 更 多 知识 。 

练习 21: (2) 修改 SimpleProxyDemo.java， 使 其 可 以 度量 方法 调用 的 次 数 。 

练习 22: (3) 修改 SimpleDynamicProxyjava。 使 其 可 以 度量 方法 调用 的 次 数 。 

练习 23: (3) 在 SimpleDynamicProxyjava 中 的 invoke0 内 部 ， 尝 试 打印 proxy 参 数 并 解释 所 
产生 的 结果 。 

， HAS. 使 用 动态 代理 来 编写 一 个 系统 以 实现 享 务 ， 其中， 代理 在 被 代理 的 调用 执行 成 功 

时 (不 抛 出 任何 异常 ) 执行 提交 ， 而 在 其 执行 失败 时 执行 回 深 。 你 的 提交 和 回 滚 都 针对 一 个 外 





O 本 书 的 英文 版 、 中 文 版 及 双语 版 均 已 由 机 械 荆 业 出 版 社 册 版 。 一 -编辑 注 
O 项 目 实际 上 是 被 用 作 术 语 项 目的 一 些 建议 ， 这 些 项 目的 解决 方案 并 未 包括 在 解决 方案 指南 中 。 
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部 的 文本 文件 ， 该 文件 不 在 Java 异 常 的 控制 范围 之 内 。 你 必须 注意 操作 的 原子 性 。 
148 SHR 


当 你 使 用 内 置 的 mall 表示 缺少 对 象 时 ， 在 每 次 使 用 引用 时 都 必须 测试 其 是 否 为 null， 这 显得 
枯燥 ,而 且 势 必 产生 相当 乏味 的 代码 。 问 题 在 于 nall 除 了 在 你 试图 用 它 执行 任何 操作 来 产生 
NullPointerException 之 外 ， 它 自己 没有 其 他 任何 行为 。 有 时 引入 空 对 象 9 的 思想 将 会 很 有 用 ， 
它 可 以 接受 传递 给 它 的 所 代表 的 对 象 的 消息 ， 但 是 将 返回 表示 为 实际 上 并 不 存在 任何 “真实 ” 
对 象 的 值 。 通 过 这 种 方式 ， 你 可 以 假设 所 有 的 对 象 都 是 有 效 的 ， 而 不 必 浪 费 编程 精力 去 检查 noll 
(并 阅读 所 产生 的 代码 )。 

尽管 想象 一 种 可 以 自动 为 我 们 创建 空 对 象 的 编程 语言 显得 很 有 趣 ， 但 是 实际 上 ， 到 处 使 用 
空 对 象 并 没有 任何 意义 一 有 时 检查 nal 就 可 以 了 ， 有 了 时 你 可 以 合理 地 假设 你 跟 本 不 会 遇 到 null， 
有 时 甚至 通过 NullPointerException 来 探测 异常 也 可 以 接受 的 。 空 对 象 最 有 用 之 处 在 于 它 更 靠近 
数据 ， 因 为 对 象 表示 的 是 问题 空间 内 的 实体 。 有 一 个 简单 的 例子 ,许多 系统 都 有 一 个 Person 类 ， 
而 在 代码 中 ， 有 很 多 情况 是 你 没有 一 个 实际 的 人 (或 者 你 有 ， 但 是 你 还 没有 这 个 人 的 全 部 信息 )， 
因此 ， 通 常 你 会 使 用 一 个 null 引 用 并 测试 它 。 与 此 不 同 的 是 ， 我 们 可 以 使 用 空 对 象 。 但 是 即使 空 
对 象 可 以 响应 “实际 ”对 象 可 以 响应 的 所 有 消息 ， 你 仍 需要 某 种 方式 去 测试 其 是 否 为 空 。 要 达 
到 此 目的 ， 最 简单 的 方式 是 创建 一 个 标记 接口 : 


1/: net/mindview/util/Null.java 
package net.mindview.util; 
public interface Null {} ///:~ 


这 使 得 instanceof 可 以 探测 空 对 象 ， 更 章 要 的 是 ， 这 并 不 要 求 你 在 所 有 的 类 中 都 添加 isNull0 
方法 (毕竟 ， 这 只 是 执行 RTTI 的 一 种 不 同方 式 一 一 为 什么 不 使 用 内 置 的 工具 呢 ?) 


//: typeinfo/Person.java 
// A Class with a Null Object. 
‘import net.mindview.util.*; 


class Person { 

public final String first; 

public final String last; ` 

public final String address; 

// etc. 

public Person(String first, String last., String address){ 
this.first = first: 
this.last = last; 
this.address = address; 


} 

public String toString() { 
return "Person: " + first +" " + last +" " + address; 

} > 

public static class NullPerson 

extends Person implements Null { 
private NullPerson() { super("None”, "None", “None"); } 
public String toString() { return “NullPerson™; } 


} 
public static final Person NULL = new NullPerson(); 
} Mi~ $ 


通常 ， 空 对 象 都 是 单 例 ， 因 此 这 里 将 其 作为 静态 final 实 例 创建 。 这 可 以 正常 工作 的 ， 因 为 
日 由 Bobby Woolf 和 Bruce Anderson 发 现 的 。 这 可 以 看 作 是 策略 模式 的 特例 。 空 对 象 的 一 种 变 体 称 为 空 过 代 器 模 


式 ， 它 使 得 在 组 合 层次 结构 中 遍历 各 个 节点 的 操作 对 客户 端 透 明 (客户 端 可 以 使 用 相同 的 逻辑 来 遍历 组 合 和 
叶子 节点 )。 = 
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Person 是 不 可 变 的 一 一 你 只 能 在 构造 器 中 设置 它 的 值 ， 然 后 读 取 这 些 值 ， 但 是 你 不 能 修改 它们 
(因为 String 自 身 具备 内 在 的 不 可 变性 )。 如 果 你 想 要 修改 一 个 NullPerson， 那 只 能 用 一 个 新 的 
了 Person 对 象 来 替换 它 。 注 意 , 你 可 以 选择 使 用 instanceof 来 探测 泛 化 的 Nullit 是 更 具体 的 NullPerson， 
但 是 由 于 使 用 了 单 例 方式 ， 所 以 你 还 可 以 只 使 用 equals0 甚 至 == 来 与 Person.Null 比 较 。 
现在 假设 你 回 到 了 互联 网 刚 出 现时 的 雄心 万 丈 的 年 代 ， 并 且 你 已 经 因 你 惊人 的 理念 而 获得 
了 一 大 笔 的 风险 投资 。 你 现在 要 招兵买马 了 ， 但 是 在 虚位以待 时 ， 你 可 以 将 Person 空 对 象 放 在 
每 个 Position 上 :， ， 
11: typeinfo/Position. java 
class Position { 
private String title: 
private Person person; 
public Position(String jobTitle, Person employee) { 
title = jobTitle; 
person = employee; 


if (person == null) 
person = Person.NULL; 


} 

public Position(String jobTitle) { 
title = jobTitle; 
person = Person. NULL; 


} 
public String getTitle() { return title; } 
i public void setTitle(String newTitle) { 
title = newTitle; 


} 
public Person getPerson() { return person: } $ 
public void setPerson(Person newPerson) { 
person = newPerson; 
if(person == nutt) 
person = Person. NULL; 


} 
public String toString() { 
return "Position: ”+ title +" ”+ person; 


} 
bh 


有 了 Position， 你 就 不 需要 创建 空 对 象 了 ， 因 为 Person.Null 的 存在 就 表示 这 是 一 个 空 Position[ 稍 
后 ， 你 可 能 会 发 现 需要 增加 一 个 显 式 的 用 于 Position 的 空 对 象 ， 但 是 YAGNI® (You Aren't Going to 
Need Ht， 你 永 不 需要 它 ) 声明 : 在 你 的 设计 草案 的 初稿 中 ， 应 该 努力 使 用 最 简单 且 可 以 工作 的 事物 ， 
直至 程序 的 某 个 方面 要 求 你 添加 额外 的 特性 ， 而 不 是 一 开始 就 假设 它 是 必需 的 ]。 

Staff 类 现在 可 以 在 你 填充 职位 时 查询 空 对 象 : 


1/: typeinfo/Staff.java 
import java.util.*; 


public class Staff extends ArrayList<Position> { 
public void add(String title, Person person) { 
add(new Position(title, person)); 
} 
public void add(String... titles) { 
for (String title : titles) 
add (new Position(title)); 


} 
public Staff(String... titles) { add(titles); } 
public boolean positionAvailable(String title) { 


O 这 是 板 限 编程 (XP) 的 原则 之 一 ， 即 “做 可 以 工作 的 最 简单 的 事情 "。 
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for (Position position : this) 
if (position. getTitle().equals(title) && 
position.getPerson() == Person.NULL) 
return true; 
return false; 
} 
public void fillPosition(String title, Person hire) {` 
for(Position position : this) | 
if (position. getTitle().equals(title) && 
position.getPerson() == Person.NULL) { 
position. setPerson(hire) ; 
return; 
} 
throw new RuntimeException( 
"Position " + title + " not available"); 


} 
Public static void main(String[] args) { 


Staff staff = new Staff("President™, "CTO", 
“Marketing Manager", "Product Manager", 
“Project Lead", “Software Engineer", 
“Software Engineer", "Software Engineer", = « 
"Software Engineer", "Test Engineer", 

"Technical Writer"); 

staff. fillPosition("President”, 
new Person("Me", "Last", “The Top, Lonely At")); 

staff.fillPosition("Project Lead", i 
new Person("Janet", "Planner", “The Burbs"); 

if (staff .positionAvailable("Software Engineer")) 
staff.fillPosition("Software Engineer”, 

new Person("Bob", "Coder", "Bright Light City")); 

System.out.printin(staff) ; 
Y 

} /* Output: $ 
(Position: President Person: Me Last The Top, Lonely At, 
Position: CTO NullPerson, Position: Marketing Manager 
NullPerson, Position: 
Project Lead Person: Janet Planner The Burbs, Position: 
Software Engineer Person: Bob Coder Bright Light City, 





Position: Software Engineer NullPerson, Position: Software 


Engineer NullPerson, Position: Software Engineer 
NullPerson, Position: Test Engineer NullPerson, Position: 
Technical Writer NullPerson) 

Whim 


Product Manager NullPerson, Position’: 


注意 ， 你 在 某 些 地 方 仍 必须 测试 空 对 象 ， 这 与 检查 是 否 为 null 没 有 差异 ， 但 是 在 其 他 地 方 
(例如 本 例 中 的 toString0 转 换 ) 你 就 不 必 执 行 额外 的 测试 了 ， 而 可 以 直接 假设 所 有 对 象 都 是 有 


效 的 


如 果 你 用 接口 取代 具体 类 ， 那 么 就 可 以 使 用 DynamicProxy 来 自动 地 创建 空 对 象 。 假 设 我 们 


有 一 个 Robot 接 口 ， 它 定义 了 一 个 名 字 、 一 个 模型 和 一 个 描述 Rbot 行 为 能 力 的 List<Operation>。 


Operation 包 含 一 个 描述 和 一 个 命令 (这 是 一 种 命令 模式 类 型 ) : 


/1/: typeinfo/Operation. java 


public interface Operation { 
String description(); a 
void command() ; 

} Mi~ 


你 可 以 通过 调用 operations0 来 访问 Robot 的 服务 : 


//: typeinfo/Robot. java 
import java.util.*; 
import net.mindview.utit.*; 


public interface Robot { 
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String name(); 
String model(); z 
List<Operation> operations(); . ' > 
class Test { 
public static void test(Robot r) { 
if(r instanceof Null) 
System.out.println(" [Null Robot] * 
System.out.printtn("Robot name: ”+ r.name()); 
System.out.printin("Robot model: " + r.model()); 
for (Operation operation : r.operations()) { 
System.out.printIn(operation.description()); 
operation. command () ; 
} 
} 
} i . 
} Mi= 


这 里 也 使 用 了 网 套 类 来 执行 测试 。 
我 们 现在 可 以 创建 一 个 扫 雪 Robot; 


//: typeinfo/SnowRemovatRobot .java 
import java.util.*; 








public class SnowRemovalRobot implements Robot { 
private String name; 
public SnowRemovalRobot (String name) {this.name = name;} 
public String name() { return name; } 
public String model() { return-"SnowBot Series 11"; } 
public List<Operation> operations() { 
return Arrays.asList( 
new Operation() { 
public String description() { 
return name + “ can shovel snow"; 
} 
Public void command() ( 
System.out.printtn(name + " shoveling snow"); 








} 
a 
new Operation() { . 

public String description() { 

return name + " can. chip ice 

} 

public void command() { 
System.out.printin(name + " chipping ice"): 





入 
}, 
new Operation() { 
public String description() { 
return name + " can clear the roof": 
} 
Public void command()-( | 
System.out.printin(name +" clearing roof"): 
} 
} 
) 
} 


public static void main(String[] args) { 
Robot .Test,test(new SnowRemovalRobot ("Slusher")): 

} 

} /* Output: 

Robot name: Slusher 

Robot model: SnowBot Series 11 

Slusher can shovel snow 

Slusher shoveling snow 

Slusher can chip ice 

Slusher chipping ice 

Slusher can clear the roof 
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Slusher clearing roof 
II 


假设 存在 许多 不 同类 型 的 Robot， 我 们 想 对 每 一 种 Robot 类 型 都 创建 一 个 空 对 象 ， 去 执行 某 
些 特殊 操作 一 一 在 本 例 中 ， 即 提供 空 对 象 所 代表 的 Robot 确 切 类 型 的 信息 。 这 些 信息 是 通过 动态 
代理 捕获 的 


/1: typeinfo/NullRobot .java 
// Using a dynamic proxy to create a Null Object. 
import java.lang.reflect.*; 

import java.util.*; 

‘import net.mindview.util.*; 





class NullRobotProxyHandler implements InvocationHandler { 
private String nullName; 7 
private Robot proxied = new NRobot(); 
NulLRobotProxyHandler(Class<? extends Robot> type) { 
nullName = type.getSimpleName() + " NullRobot™; Fag 
} 
Private class NRobot implements Null, Robot { 
public String name() { return nullName; } 
public String model() { return nullNam 
public List<Operation> operations() { 
return Collections,emptyList(); 
} 





} 


} 
public Object 
tnvoke (Object proxy, Method method, Object[] args) 
throws Throwable { 1 
return method. invoke(proxied, args); 
} 
} 


public class NullRobot { 
public static Robot 
NewNullRobot (Class<? extends Robot> type) { 
return (Robot)Proxy.newProxyInstance( ia ` 
NullRobot.class.getClassLoader(), 
new Class{]{ Null.class, Robot.class }, 
new NulLRobotProxyHandler(type)); 
} 
public static void main(String[] args) { 
Robot[] bots = { 
new SnowRemovalRobot ("SnowBee") , 
newNullRobot (SnowRemovalRobot .class) 
}; 
for (Robot bot : bots) 
Robot .Test.test (bot) ; 
} 


} /* Output: 
Robot name: SnowBee 

Robot model: SnowBot Series 11 

SnowBee can shovel snow 

SnowBee shoveling snow 3 
SnowBee can chip ice 

SnowBee chipping ice 

SnowBee can clear the roof 

SnowBee clearing roof | R 

[Null Robot] s 

Robot name: SnowRemovalRobot NullRobot 

Robot model: SnowRemovalRobot NullRobot 

li~ a: - 


无 论 何 时 ， 如 果 你 需要 一 个 空 Robot 对 象 ， 只 需 调 用 newNullRobot0 ， 并 传递 需要 代理 的 
Robot 的 类 型 。 代 理会 满足 Robot 和 Null 接 口 的 需求 ， 并 提供 它 所 代理 的 类 型 的 确切 名 字 。 + 
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14.8.1 模拟 对 象 与 桩 

空 对 象 的 逻辑 变 体 是 模拟 对 象 和 桩 。 与 空 对 象 一 样 ， 它 们 都 表示 在 最 终 的 程序 中 所 使 用 的 
“实际 ”对 象 。 但 是 ， 模 拟 对 象 和 桩 都 只 是 假扮 可 以 传递 实际 信息 的 存活 对 象 ， 而 不 是 像 空 对 象 
那样 可 以 成 为 null 的 一 种 更 加 智能 化 的 替代 物 。 

模拟 对 象 和 柱 之 间 的 差异 在 于 程度 不 同 。 模 拟 对 象 往往 是 轻 量 级 和 自 测试 的 ， 通 常 很 多 模 
拟 对 象 被 创建 出 来 是 为 了 处 理 各 种 不 同 的 测试 情况 。 桩 只 是 返回 桂 数 据 ， 它 通常 是 重量 级 的 ， 
并 且 经 常 在 测试 之 间 被 复 用 。 桩 可 以 根据 它们 被 调用 的 方式 ， 通 过 配置 进行 修改 ， 因 此 桩 是 一 
种 复杂 对 象 ， 它 要 做 很 多 事 。 然 而 对 于 模拟 对 象 ， 如 果 你 需要 做 很 多 事情 ， 通 常会 创建 大 量 小 
而 简单 的 模拟 对 象 。 

[606] 练习 24，(4) 在 RegisteredFactoriesjava 中 添加 空 对 象 。 


14.9 接口 与 类 型 信息 


interface 关 键 字 的 一 种 重要 目标 就 是 允许 程序 员 隔 离 构 件 ， 进 而 降低 看 合 性 。 如 果 你 编写 
接口 ， 那 么 就 可 以 实现 这 一 目标 ， 但 是 通过 类 型 信息 ， 这 种 耦合 性 还 是 会 传播 出 去 ~ 一 接口 并 
非 是 对 解 而 的 一 种 无 懈 可 击 的 保障 。 下 面 有 一 个 示例 ， 先 是 一 个 接口 : 


/1/: typeinfo/ interfacea/A. fava 
package typeinfo. interfacea; 





public interface A { 
void fO; 
} Mi~ 


然后 实现 这 个 接口 ， 你 可 以 看 到 其 代码 是 如 何 围绕 着 实际 的 实现 类 型 潜行 的 : 


//:_typeinfo/InterfaceViolation. java 
// Sneaking around an interface. 
import typeinfo. interfacea.*; 


class B implements A { 
public void f() {} 
public void g() {} 

$ 


public class InterfaceViolation { 
public static void main(String[] args) { 
A a = new B(); 
af(); 
// a.gQ); // Compile error 
System.out.printin(a.getClass().getName()); 
if(a instanceof B) { 
B b = (B)a; 
b.gQ: 
} 


} 

} /* Output: 

B 

SHI: 

通过 使 用 RTTI， 我 们 发 现 a 是 被 当 作 B 实 现 的 。 通 过 将 其 转型 为 B， 我 们 可 以 调用 不 在 A 中 的 
方法 。 : 
这 是 完全 合法 和 可 接受 的 ， 但 是 你 也 许 并 不 想 让 客户 端 程序 员 这 么 做 ， 因 为 这 给 了 他 们 一 
个 机 会 ， 使 得 他 们 的 代码 与 你 的 代 玛 的 耦合 程度 超过 你 的 期 望 。 也 就 是 说 ， 你 可 能 认为 
interface 关 键 字 正在 保护 着 你 ， 但 是 它 并 没有 ， 在 本 例 中 使 用 B 来 实现 A 这 一 事实 是 公开 有 案 , 
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可 查 的 。 

一 种 解决 方案 是 直接 声明 ， 如 果 程序 员 决 定 使 用 实际 的 类 而 不 是 接口 ， 他 们 需要 自己 对 自 
己 负责 。 这 在 很 多 情况 下 可 能 都 是 合理 的 ， 但 “可 能 ”还 不 够 ， 你 也 许 希望 应 用 一 些 更 严 苛 的 
控制 。 

最 简单 的 方式 是 对 实现 使 用 包 访问 权限 ， 这 样 在 包 外 部 的 客户 端 就 不 能 看 到 它 了 : 


1/: typeinfo/packageaccess/HiddenC. java 
package typeinfo.packageaccess; 

import typeinfo. interfacea.*; 

import static net.mindview.util.Print.*; 


class C implements A { 
public void f() { print("public C.f()2); } 
public void g() { print("public C.g()"); } 
void u() { print("package C.u()"); } 
protected void v() { print("protected C.v()"); } 
private void wO { print("private C.w()"); } 

$ 


public class Hiddenc { 
public static A makeA() { return new C(); } 
} Mi~ 


在 这 个 包 中 唯一 public 的 部 分 ， 即 HiddenC， 在 被 调用 时 将 产生 A 接 口 类 型 的 对 象 。 这 里 有 趣 之 
处 在 于 : 即使 你 从 makeAO 返 回 的 是 C 类 型 ， 你 在 包 的 外 部 仍旧 不 能 使 用 A 之 外 的 任何 方法 ， 因 
为 你 不 能 在 包 的 外 部 命名 C。 

现在 如 果 你 试图 将 其 向 下 转型 为 C， 则 将 被 禁止 ， 因 为 在 包 的 外 部 没有 任何 C 类 型 可 用 ， 


//:_typeinfo/HiddenImplementation. java 
// Sneaking around package access. 
import typeinfo. interfacea.*; 

import typeinfo.packageacces 
import java. lang. reflect. *; 








public class HiddenImplementation { 
public static void main(String[] args) throws Exception { 
A a = HiddenC.makeA() ; 
af; 
System.out.printin(a.getClass().getName()); 
// Compite error: cannot find symbol 'C': 
/* if(a instanceof C) { 
Ce= (a; 
c80; 
Fe 
// Oops! Reflection still allows us to call g0: 
callHiddentethod(a, "g"); 
// And even methods that are less accessible! 
callHiddentethod(a, "u"); 
callHiddenMethod(a, "v" 
callHiddentethod(a, 
} 
static void callHiddenMethod (Object a, String methodName) 
throws Exception { 
Method g = a.getClass().getDeclaredMethod (methodName) ; 
&.setAccessible(true) ; 
g. invoke(a) ; 











O 这 种 情况 最 出 名 的 案例 就 是 Windows 操 作 系统 ， 它 有 一 个 发 布 的 API， 并 假设 你 会 对 着 它 进 行 编码 ， 另 外 还 有 
一 个 未 发 布 的 ， 但 是 可 视 的 函数 集 ， 你 可 以 发 现 并 调用 它 。 为 了 解决 问题 ， 程 序 员 会 使 用 隐藏 的 API 函 数 ， 这 
导致 微软 必须 把 它们 当 作 公共 API! 的 一 部 分 来 维护 。 因 而 成 为 了 公司 巨额 成 本 和 投入 的 黑洞 。 
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} /* Output: 
public C.f() 
type info. packageaccess.C s ; ir 
public C.g() 

+ package C.u() 
protected C.v() 
private C.w() 
Whim 1 


正如 你 所 看 到 的 ， 通 过 使 用 反射 ， 仍 旧 可 以 到 达 并 调用 所 有 方法 ， 甚 至 是 private 方 法 ! 如 果 知 








[609] 道 方法 名 ， 你 就 可 以 在 其 Method 对 象 上 调用 setAccessible(true)， 就 像 在 caliHiddenMethod0 中 








看 到 的 那样 。 
你 可 能 会 认为 ， 可 以 通过 只 发 布 编译 后 的 代码 来 阻止 这 种 情况 ， 但 是 这 并 不 解决 问题 。 因 为 
只 需 运行 javap， 一 个 随 JDK 发 布 的 反 编译 器 即 可 突破 这 一 限制 。 下 面 是 一 个 使 用 它 的 命令 行 ; 
javap -private C 
-private 标 志 表示 所 有 的 成 员 都 应 该 显示 ， 其 至 包括 私有 成 员 。 下 面 是 输出 ， 。 


class typeinfo.packageaccess.C extends 
java.lang.Object implements typeinfo.interfacea.A { 
typeinfo. packageaccess.C(); 
public void fO; 
public void g(); 
void uC); , 
protected void v(); 
private void wO); 
} 


因此 任何 人 都 可 以 获取 你 最 私有 的 方法 的 名 字 和 签名 ， 然 后 调用 它们 。 
如 果 你 将 接口 实现 为 一 个 私有 内 部 类 ， 又 会 怎样 呢 ? 下 面 展 示 了 这 种 情况 ，， 


//: typeinfo/InnerImplementation. java x 
// Private inner classes can't hide from reflection. 

import typeinto. interfacea.*; A 
import static net.mindview.util.Print.*; 


ctass InnerA { 
private static class C implements A { 
public void f() { print("public C.f()"); } 
public void g() { print("public C.g()"); } 
void u() { print ("package C.u()"); } 
protected void v() { print("protected C.v()"); } 
private void w() { print("private C.w()"); } 


} 
public static A makeA() { return new C(); } 








610) public class InnerImplementation { i 
public static void main(String[] args) throws Exception { 

A a = InnerA.makeA(); 
atO; mat 
System. out.printin(a.getClass() .getName()): 
// Reflection still gets into the private class: ‘ 
HiddenImplementation.callHiddenMethod(a, 
HiddenImplementation.callHiddenMethod(a, 
HiddenImplementation.callHiddenMethod(a, 
HiddenImplementation.callHiddenMethod(a, 











} /* Output: 

public C.f() 

InnerASC g 
public C.g() 

package C.u() 

protected C.v.() 
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private C.w() 
Whim 


这 里 对 反射 仍旧 没有 隐藏 任何 东西 。 那 么 如 果 是 匿名 类 呢 ? 


/17: typeinfo/AnonymousImplementation. java 
// Anonymous inner classes can't hide from reflection 
import typeinfo. interfacea.*; 

import static net.mindview.util.Print.*; 


class AnonymousA { 
public static A makeA() { 
return new AQ { 
public void f() { print("public C.f()"): } 
public void g() { print("public C.g()"); } 
void u() { print("package C.u()"); } 
protected void v() { print("protected C.v()"); } 
private void w() { print("private C.w()"); } 
ye 
} 
} 





public class AnonymousImplementation { 
public static void main(String[] args) throws Exception { 
A a = AnonymousA.makeA() ; 
af: 
System.out.printIn(a.getClass().getName()); 
// Reflection still gets into the anonymous class: 
HiddenImplementation.callHiddenMethod(a, “g"); 
HiddenImplementation.callHiddenMethod(a, "u" 
HiddenImplementation.callHiddenMethod(a, "v" 
HiddenImplementation.callHiddenMethod(a, "w"); 
} 
} /* Output: 
public C.f() 
‘AnonymousAS1 
public C.g() 
package C.u() 
protected C.v() 
private C.w() 
Whim 


看 起 来 没有 任何 方式 可 以 阻止 反射 到 达 并 调用 那些 非 公 共 访 问 权限 的 方法 。 对 于 域 来 说 ， 






的 确 如 此 ， 即 便 是 private 域 : 


1/: typeinfo/Modi fyingPrivateFields. java 
import java. lang.reflect.*; 


class WithPrivateFinalField { 
private int 1 = 1; 
private final String s = "I'm totally safe"; 
private String s2 = “Am I safe?"; - 
public String toString() { 
return "i= "+44", "+54", "+ 52; 
} 
} 


public class ModifyingPrivateFields { 

public static void main(String{] args) throws Exception { 
WithPrivateFinalField pf = new WithPrivateFinalField(); 
System. out.printin(pf 
Field f = pf.getClass().getDeclaredField("i"); 
f.setAccessible(true) ; 
System.out.printin("f.getint(pf): ”+ f.getInt(pf)): 
f.setInt(pf, 47); 
System. out.printIn(pf); 
f = pf.getClass().getDeclaredField("s"); 
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ay 
四 


f.setAccessible(true); 
System.out.printin("f.get(pf): ”+ f.get(pf)); 
f.set(pf, "No, you're not!"); 
System.out.printin(pf) ; 

f = pf.getClass() .getDeclaredField("s2"); 
f.setAccessible(true) ; 
System.out.printin("f.get (pf): ”+ f.get(pf)); 
f.set(pf, "No, you're not!"); 

System. out.printin(pf) ; 


} 
} /* Output: 
i = 1, I'm totally safe, Am I safe? 
f.getInt(pf): 1 
i = 47, I'm totally safe, Am I safe? 
f.get(pf): I'm totally safe 
i = 47, I'm totally safe, Am I safe? 
f.get(pf): Am I safe? 
1 = 47, I'm totally safe, No, you're not! 
Whim 


但 是 ，final 域 实际 上 在 遭遇 修改 时 是 安全 的 。 运 行 时 系统 会 在 不 抛 异常 的 情况 下 接受 任何 修改 
尝试 ,但 是 实际 上 不 会 发 生 任何 修改 。 

通常 ， 所 有 这 些 违反 访问 权限 的 操作 并 非 世上 最 遭 之 事 。 如 果 有 人 使 用 这 样 的 技术 去 调用 
标识 为 private 或 包 访问 权限 的 方法 (很 明显 这 些 访问 权限 表示 这 些 人 不 应 该 调用 它们 )， 那 么 对 
他 们 来 说 ， 如 果 你 修改 了 这 些 方法 的 某 些 方面 ， 他 们 不 应 该 抱怨 。 另 一 方面 ， 总 是 在 类 中 留 下 
后 门 的 这 一 事实 ， 也 许可 以 使 得 你 能 够 解决 某 些 特定 类 型 的 问题 ， 但 如 果 不 这 样 做 ， 这 些 问题 
将 难以 或 者 不 可 能 解决 ， 通 常 反射 带 来 的 好 处 是 不 可 否认 的 。 

练习 25: (2) 创建 一 个 包含 private、protected 和 包 访问 权限 方法 的 类 ， 编 写 代码 在 该 类 所 处 
的 包 的 外 部 调用 访问 这 些 方法 。 


14.10 总 结 


RTTI 人 允许 通 过 匿名 基 类 的 引用 来 发 现 类 型 信息 。 初 学 者 极 易 误 用 它 ， 因 为 在 学 会 使 用 多 态 
调用 方法 之 前 ， 这 么 做 也 很 有 效 。 对 有 过 程 化 编程 背景 的 人 来 说 ， 很 难 让 他 们 不 把 程序 组 织 成 
一 系列 switeh 语 句 。 你 可 以 用 RTTI 做 到 这 一 点 ， 但 是 这 样 就 在 代码 开发 和 维护 过 程 中 损失 了 多 
态 机 制 的 重要 价值 。 面 向 对 象 编程 语言 的 目的 是 让 我 们 在 凡是 可 以 使 用 的 地 方 都 使 用 多 坊 机 制 ， 
只 在 必需 的 时 候 使 用 RTTI。 

然而 使 用 多 态 机 制 的 方法 调用 ， 要 求 我 们 拥有 基 类 定义 的 控制 权 ， 因 为 在 你 扩展 程序 的 时 
候 ， 可 能 会 发 现 基 类 并 未 包含 我 们 想 要 的 方法 。 如 果 基 类 来 别人 的 类 ， 或 者 由 别人 控制 ， 这 时 
候 RTTI 便 是 一 种 解决 之 道 :可 继承 一 个 新 类 ， 然 后 添加 你 需要 的 方法 。 在 代码 的 其 他 地 方 ， 可 
以 检查 你 自己 特定 的 类 型 ， 并 调用 你 自己 的 方法 。 这 样 做 不 会 破坏 多 态 性 以 及 程序 的 扩展 能 力 ， 
因为 这 样 添加 一 个 新 的 类 并 不 需要 在 程序 中 搜索 switch 语 句 。 但 如 果 在 程序 主体 中 添加 需要 的 新 
特性 的 代码 ， 就 必须 使 用 RTTI 来 检查 你 的 特定 的 类 型 。 

如 果 只 是 为 了 某 个 特定 类 的 利益 ， 而 将 某 个 特性 放 进 基 类 里 ， 这 意味 着 从 那个 基 类 派生 出 
的 所 有 其 他 子 类 都 带 有 这 些 可 能 无 意义 的 东西 。 这 会 使 得 接口 更 不 清晰 ， 因 为 我 们 必须 履 盖 由 
基 类 继承 而 来 的 所 有 抽象 方法 ， 这 是 很 恼人 的 。 例 如 ， 考 虑 一 个 表示 乐器 Instrument 的 类 层次 
结构 。 假 设 我 们 想 清洁 管弦 乐队 中 某 些 乐器 残留 的 口水 ， 一 种 办 法 是 在 基 类 Instrument 中 放 入 
clearSpitValve0 方 法 。 但 这 样 做 会 造成 混淆 ， 因 为 它 意 味 着 打击 乐器 Percussion、 弦 乐器 
Stringed 和 电子 乐器 Electronic 也 需要 清洁 口水 。 在 这 个 例子 中 ，RTTI 可 以 提供 了 一 种 更 合理 的 
解决 方案 。 可 以 将 clearSpitValve0 置 于 适当 的 特定 类 中 ,在 这 个 例子 中 是 Wind (管乐器 )。 同 时 ， 
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你 可 能 会 发 现 还 有 更 恰当 的 解决 方法 ， 在 这 里 ， 就 是 将 prepareInstrumentO 置 于 基 类 中 ， 但 是 
初次 面 对 这 个 问题 时 读者 可 能 看 不 到 这 样 的 解决 方案 ， 而 误 认为 必须 使 用 RTTI。 

最 后 一 点 ，RTTI 有 时 能 解决 效率 问题 。 也 许 你 的 程序 漂亮 地 运用 了 多 态 ， 但 其 中 某 个 对 象 
是 以 极端 缺乏 效率 的 方式 达到 这 个 目的 的 。 你 可 以 挑 出 这 个 类 ， 使 用 RTTI， 并 且 为 其 编写 一 段 
特别 的 代码 以 提高 效率 。 然 而 必须 要 注意 ， 不 要 太 早 地 关注 程序 的 效率 问题 ， 这 是 个 诱 人 的 陷 
阱 。 最 好 首先 让 程序 运作 起 来 ， 然 后 再 考虑 它 的 速度 ， 如 果 要 解决 效率 问题 可 以 使 用 profiler 
(查看 http://MindView.net/Books/BetterJava 上 的 补充 材料 )。 

我 们 已 经 看 到 了 ， 由 于 反射 允许 更 加 动态 的 编程 风格 ， 因 此 它 开创 了 编程 的 新 世界 。 对 有 
些 人 来 说 ， 反 射 的 动态 特性 是 一 种 烦 扰 ， 对 于 已 经 习惯 于 静态 类 型 检查 的 安全 性 的 人 来 说 ， 你 
可 以 执行 一 些 只 能 在 运行 时 进行 的 检查 ， 并 用 异常 来 报告 检查 结果 的 行为 ， 这 本 身 就 是 一 种 错 
误 的 方向 。 有些 人 走 的 更 远 ， 他 们 声称 引入 运行 时 异常 本 身 就 是 一 种 指示 ， 说 明 应 该 避免 这 种 
代码 。 我 发 现 这 种 意义 的 安全 是 一 种 错觉 ， 因 为 总 是 有 些 事情 是 在 运行 时 发 生 并 抛 出 异常 的 ， 
即使 是 在 不 包含 任何 try 语 句 块 或 异常 规格 说 明 的 程序 中 也 是 如 此 。 因 此 ， 我 认为 一 致 的 错误 报 
告 模型 的 存在 使 我 们 能 够 通过 使 用 反射 编写 动态 代码 。 当 然 ， 尽 力 编写 能 够 进行 静态 检查 的 代 
码 是 值得 的 ， 只 要 你 确实 能 够 这 么 做 。 但 是 我 相信 动态 代码 是 将 Java 与 其 他 诸如 C++ 这 样 的 语言 
区 分 开 的 重要 工具 之 一 。 

练习 26: (3) 实现 本 章 总 结 中 所 描述 的 clearSpitValve0 。 

所 选 习 题 的 答案 都 可 以 在 名 为 The Thinking in Java Annotated Solution Guide 的 电子 文档 中 找 
到 ， 读 者 可 以 从 wwwMindView.net 购 买 此 文档 。 
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第 15 章 Z Æ 


一 般 的 类 和 方法 ， 只 能 使 用 具体 的 类 型 ， 要 么 是 基本 类 型 ， 要 么 是 自 定义 的 类 。 如 果 要 编 
写 可 以 应 用 于 多 种 类 型 的 代码 ， 这 种 刻板 的 限制 对 代码 的 束缚 就 会 很 大 ” 。 

在 面向 对 象 编程 语言 中 ， 多 态 算是 一 种 泛 化 机 制 。 例 如 ， 你 可 以 将 方法 的 参数 类 型 设 为 基 
类 ， 那 么 该 方法 就 可 以 接受 从 这 个 基 类 中 导出 的 任何 类 作为 参数 。 这 样 的 方法 更 加 通用 一 些 ， 
可 应 用 的 地 方 也 多 一 些 。 在 类 的 内 部 也 是 如 此 ， 凡 是 需要 说 明 类 型 的 地 方 ， 如 果 都 使 用 基 类 
确实 能 够 具备 更 好 的 灵活 性 。 但 是 ， 考 虑 到 除了 final 类 。 不 能 扩展 ， 其 他 任何 类 都 可 以 被 扩展 ， 
所 以 这 种 灵活 性 大 多 数 时 候 也 会 有 一 些 性 能 损耗 

有 了 时候， 拘泥 于 单 继承 体系 ， 也 会 使 程序 受 限 太 多 。 如 果 方 法 的 参数 是 一 个 接口 ， 而 不 是 
一 个 类 ， 这 种 限制 就 放松 了 许多 。 因 为 任何 实现 了 该 接口 的 类 都 能 够 满足 该 方法 ， 这 也 包括 暂 
时 还 不 存在 的 类 。 这 就 给 予 客户 端 程序 员 一 种 选择 ， 他 可 以 通过 实现 一 个 接口 来 满足 类 或 方法 。 
因此 ， 接 口 允许 我 们 快捷 地 实现 类 继承 ， 也 使 我 们 有 机 会 创建 一 个 新 类 来 做 到 这 一 点 。 

可 是 有 的 时 候 ， 即 便 使 用 了 接口 ， 对 程序 的 约束 也 还 是 太 强 了 。 因 为 一 旦 指明 了 接口 ， 它 
就 要 求 你 的 代码 必须 使 用 特定 的 接口 。 而 我 们 希望 达到 的 目的 是 编写 更 通用 的 代码 ， 要 使 代码 
能 够 应 用 于 “ 某 种 不 具体 的 类 型 ”， 而 不 是 一 个 具体 的 接口 或 类 。 

这 就 是 Java SE5 的 重大 变化 之 一 : 泛 型 的 概念 。 泛 型 实现 了 参数 化 类 型 的 概念 ， 使 代码 可 以 
应 用 于 多 种 类 型 。“ 泛 型 ”这 个 术语 的 意思 是 :“ 适 用 于 许多 许多 的 类 型 "。 泛 型 在 编程 语言 中 出 
现时 ， 其 最 初 的 目的 是 希望 类 或 方法 能 够 具备 最 广泛 的 表达 能 力 。 如 何 做 到 这 一 点 呢 ， 正 是 通 
过 解 艳 类 或 方法 与 所 使 用 的 类 型 之 间 的 约束 。 稍 后 你 将 看 到 ，Java 中 的 泛 型 并 没有 这 么 高 的 扎 
求 ， 实 际 上 ， 你 可 能 会 质疑 ，Java 中 的 术语 “ 泛 型 ”是 否 适合 用 来 描述 这 一 功能 。 

如 果 你 从 未 接触 过 参数 化 类 型 机 制 ， 那 么 ， 在 学 习 了 Java 中 的 泛 型 之 后 ， 你 会 发 现 ， 对 这 
门 语言 而 言 ， 泛 型 确实 是 一 个 很 有 益 的 补充 。 在 你 创建 参数 化 类 型 的 一 个 实例 时 ， 编 译 器 会 为 
你 负责 转型 操作 ， 并 且 保证 类 型 的 正确 性 。 这 应 该 是 一 个 进步。 

然而 ， 如 果 你 了 解 其 他 语言 (例如 C++) 中 的 参数 化 类 型 机 制 ， 你 就 会 发 现 ， 有 些 以 前 能 
做 到 的 事情 ， 使 用 Java 的 泛 型 机 制 却 无 法 做 到 。 使 用 别人 已 经 构建 好 的 泛 型 类 型 会 相当 容易 。 
但 是 如 果 你 要 自己 创建 一 个 泛 型 实例 ， 就 会 遇 到 许多 令 你 吃惊 的 事情 。 在 本 章 中 ， 我 的 任务 之 
一 就 是 向 你 解释 ，Java 中 的 泛 型 是 怎样 发 展 成 现在 这 样 的 。 

这 并 非 是 说 Java 的 泛 型 毫 无 用 处 。 在 很 多 情况 下 ， 它 们 可 以 使 代码 更 直接 更 优雅 。 不 过 ， 如 
果 你 具备 其 他 语言 的 经 验 ， 而 那 种 语言 实现 了 更 纯粹 的 泛 型 ， 那 么 ，Java 可 能 令 你 失望 了 。 在 本 
章 中 ， 我 们 会 介绍 Java 泛 型 的 优点 与 局 限 ， 希 望 这 能 够 帮助 你 更 有 效 地 使 用 Java 的 这 个 新 功能 。 


15.1 与 C++ 的 比较 


Java 的 设计 者 曾 说 过 ， 设 计 这 门 语言 的 灵感 主要 来 自 C++。 尽 管 如 此 ， 学 习 Java 时 ， 基 本 上 


O 在 我 写作 本 章 时 ，Angelika Langer 的 《Java Generics FAQ) (2 I www.angelikalanger.com/GenericsFAQ/ 
JavaGenericsFAQ html) 以 及 她 《与 Klaus Kreft 合 著 ) 的 另 一 本 著作 给 予 了 我 很 大 的 帮助 。 
O 或 者 具有 private 构 造 器 的 类 。 
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可 以 不 用 参考 C++。 我 也 是 尽力 这 样 做 的 ， 除 非 ， 与 C++ 的 比较 能 够 加 深 你 的 理解 。 
Java 中 的 泛 型 就 需要 与 C++ 进行 一 番 比较 ， 理 由 有 二 : 首先 ， 了 解 C++ 模板 的 某 些 方面 ， 有 
助 于 你 理解 泛 型 的 基础 。 同 时 ， 非 常 重 要 的 一 点 是 ， 你 可 以 了 解 Java 泛 型 的 局 限 是 什么 ,以 及 
为 什么 会 有 这 些 限制 。 最 终 的 目的 是 帮助 你 理解 ，Java 泛 型 的 边界 在 哪里 。 根 据 我 的 经 验 ， 理 [6l8 
解 了 边界 所 在 ， 你 才能 成 为 程序 高 手 。 因 为 只 有 知道 了 某 个 技术 不 能 做 到 什么 ， 你 才能 更 好 地 
做 到 所 能 做 的 (部 分 原因 是 ， 不 必 浪 费时 间 在 死胡同 里 乱 转 )。 
第 二 个 原因 是 ,在 Java 社 区 中 ， 人 们 普遍 对 C++ 模板 有 一 种 误解 ， 而 这 种 误解 可 能 会 误导 你 ， 
令 你 在 理解 泛 型 的 意图 时 产生 偏差 。 
因此 ， 在 本 章 中 会 介绍 一 些 C++ 模板 的 例子 ， 不 过 我 也 会 尽量 控制 它们 的 篇 幅 。 


15.2 简单 泛 型 


有 许多 原因 促成 了 泛 型 的 出 现 ， 而 最 引 人 注 目的 一 个 原因 ， 就 是 为 了 创造 容器 类 。( 关 于 容 
器 类 ， 你 可 以 参考 第 11 章 和 第 17 章 这 两 章 。) 容器 ， 就 是 存放 要 使 用 的 对 象 的 地 方 。 数 组 也 是 如 
此 ， 不 过 与 简单 的 数组 相 比 ， 容 器 类 更 加 灵活 ， 具 备 更 多 不 同 的 功能 。 事 实 上 ， 所 有 的 程序 ， 
在 运行 时 都 要 求 你 持 有 一 大 堆 对 象 ， 所 以 ， 容 器 类 算得 上 最 具 重用 性 的 类 库 之 一 。 

我 们 先 来 看 看 一 个 只 能 持 有 单个 对 象 的 类 。 当 然 了 ， 这 个 类 可 以 明确 指定 其 持 有 的 对 象 的 
类 型 ， 


/1/: generics/Hotderl.java 














class Automobile {} 


public class Hotderl { 
private Automobile a; 
public Holderl(Automobile a) { this.a = a; } 
Automobile get() { return a; } 
} Mi~ 
不 过 ， 这 个 类 的 可 重用 性 就 不 怎么 样 了 ， 它 无 法 持 有 其 他 类 型 的 任何 对 象 。 我 们 可 不 希望 
为 碰 到 的 每 个 类 型 都 编写 一 个 新 的 类 。 
在 Java SE5 之 前 ， 我 们 可 以 让 这 个 类 直接 持 有 Object 类 型 的 对 象 ， 


/1/: generics/Holder2. java 


public class Holder2 { =— 
private Object a; 
public Holder2(Object a) { this.a = a; } 


| 
S| 








[ 


public void set(Object a) { this.a = a; } 
public Object get() { return a; } 
public static void main(String[] args) { 
Holder2 h2 = new Holder2(new Automobile()); 
Automobile a = (Automobile)h2.get(); 
h2.set("Not an Automobile"); 
String s = (String)h2.get(); 
h2.set(1); // Autoboxes to Integer 
Integer x = (Integer)h2.get(); 


} 
} i~ 
现在 ，Holder2 可 以 存储 任何 类 型 的 对 象 ， 在 这 个 例子 中 ， 只 用 了 一 个 Holder2 对 象 ， 却 先 
后 三 次 存储 了 三 种 不 同类 型 的 对 象 。 
有 些 情况 下 ， 我 们 确实 希望 容器 能 够 同时 持 有 多 种 类 型 的 对 象 。 但 是 ， 通 常 而 言 ， 我 们 只 
会 使 用 容器 来 存储 一 种 类 型 的 对 象 。 泛 型 的 主要 目的 之 一 就 是 用 来 指定 容器 要 持 有 什么 类 型 的 
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对 象 ， 而 且 由 编译 器 来 保证 类 型 的 正确 性 。 

因此 ， 与 其 使 用 Object， 我 们 更 喜欢 暂时 不 指定 类 型 ， 而 是 稍 后 再 决定 具体 使 用 什么 类 型 。 
要 达到 这 个 目的 ， 需 要 使 用 类 型 参数 ， 用 尖 括 号 括 住 ， 放 在 类 名 后 面 。 然 后 在 使 用 这 个 类 的 时 
候 ， 再 用 实际 的 类 型 替换 此 类 型 参数 。 在 下 面 的 例子 中 ，T 就 是 类 型 参数 ， 


//: generics/Holder3.java 


public class Holder3<T> { 
private T a; 
public Holder3(T a) { this.a = a; } 
public void set(T a) { this.a = a; } 
public T get() { return a; } 
public static void main(String[] args) { 
Holder3<Automobile> h3 = 
new Holder3<Automobile>(new Automobile()); 
Automobile a = h3.get(); // No cast needed 
// h3.set("Not an Automobile"); // Error 
/1 h3.set(1); // Error 


} 
} M~ 


现在 ， 当 你 创建 Holder3 对 象 时 ， 必 须 指明 想 持 有 什么 类 型 的 对 象 ， 将 其 置 于 尖 括 号 内 。 就 
像 main0 中 那样 。 然 后 ， 你 就 只 能 在 Holder3 中 存 人 该 类 型 (或 其 子 类 ， 因 为 多 态 与 泛 型 不 冲突 ) 
的 对 象 了 。 并 且 ， 在 你 从 Holder3 中 取出 它 持 有 的 对 象 时 ， 自 动 地 就 是 正确 的 类 型 。 

这 就 是 Java 泛 型 的 核心 概念 ， 告 诉 编译 器 想 使 用 什么 类 型 ， 然 后 编译 器 帮 你 处 理 一 切 细节 。 

一 般 而 言 ， 你 可 以 认为 泛 型 与 其 他 的 类 型 差不多 ， 只 不 过 它们 磁 巧 有 类 型 参数 罢了 。 稍 后 
我 们 会 看 到 ， 在 使 用 泛 型 时 ， 我 们 只 需 指定 它们 的 名 称 以 及 类 型 参数 列表 即 可 。 

练习 1: (1) 配合 typeinfo.pets 类 库 ， 用 Holder3 来 证 明 ， 如 果 指 定 Holder3 可 以 持 有 某 个 基 类 
类 型 ， 那 么 它 也 能 持 有 导出 类 型 。 

练习 2: (1) 创建 一 个 Holder 类 ， 使 其 能 够 持 有 具有 相同 类 型 的 3 个 对 象 ， 并 提供 相应 的 读 写 
方法 访问 这 些 对 象 ， 以 及 一 个 可 以 初始 化 其 持 有 的 3 个 对 象 的 构造 器 
15.2.1 一 个 元 组 类 库 

仅 一 次 方法 调用 就 能 返回 多 个 对 象 ， 你 应 该 经 常 需 要 这 样 的 功能 吧 。 可 是 return 语 句 只 允许 
返回 单个 对 象 ， 因 此 ， 解 决 办 法 就 是 创建 一 个 对 象 ， 用 它 来 持 有 想 要 返回 的 多 个 对 象 。 当 然 ， 
可 以 在 每 次 需要 的 时 候 ， 专 门 创建 一 个 类 来 完成 这 样 的 工作 。 可 是 有 了 泛 型 ， 我 们 就 能 够 一 次 
性 地 解决 该 问题 ， 以 后 再 也 不 用 在 这 个 问题 上 浪费 时 间 了 。 同 时 ， 我 们 在 编译 期 就 能 确保 类 型 
安全 。 

这 个 概念 称 为 元 组 (tuple) ， 它 是 将 一 组 对 象 直接 打包 存储 于 其 中 的 一 个 单一 对 象 。 这 个 容 
器 对 象 允 许 读 取 其 中 元 素 ， 但 是 不 允许 向 其 中 存放 新 的 对 象 。( 这 个 概念 也 称 为 数据 传送 对 象 ， 
或 信使 。) 

通常 ， 元 组 可 以 具有 任意 长 度 ， 同 时 ， 元 组 中 的 对 象 可 以 是 任意 不 同 的 类 型 。 不 过 ， 我 们 
希望 能 够 为 每 一 个 对 象 指明 其 类 型 ， 并 且 从 容器 中 读 取出 来 时 ， 能 够 得 到 正确 的 类 型 。 要 处 理 
不 同 长 度 的 问题 ， 我 们 需要 创建 多 个 不 同 的 元 组 。 下 面 的 程序 是 一 个 2 维 元 组 ， 它 能 够 持 有 两 个 
WR: 


//: net/mindview/util/TwoTuple. java 
package net.mindview.util; 
public class TwoTuple<A,B> { 
public final A first; 
public final B second; 
Public TwoTuple(A a, B b) { first = a; second = b; } 
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public String toString() { 
return "(" + first +", "+ second + ")"; 


) 

} i~ 

构造 器 捕获 了 要 存储 的 对 象 ， 而 toString0 是 一 个 便利 函数 ， 用 来 显示 列表 中 的 值 。 注 意 ， 
元 组 隐 含 地 保持 了 其 中 元 素 的 次 序 。 

第 一 次 阅读 上 面 的 代码 时 ， 你 也 许 会 想 ， 这 不 是 违反 了 Java 编 程 的 安全 性 原则 吗 ? first 和 
second 应 该 声明 为 private， 然 后 提供 getFirst0 和 getSecond0 之 类 的 访问 方法 才 对 呀 ?让 我 们 仔 
细 看 看 这 个 例子 中 的 安全 性 :客户 端 程序 可 以 读 取 first 和 second 对 象 ， 然 后 可 以 随心 所 欲 地 使 用 
这 两 个 对 象 。 但 是 ， 它 们 却 无 法 将 其 他 值 赋 对 first 或 second。 因 为 final 声 明 为 你 买 了 相同 的 安全 
保险 ,而且 这 种 格式 更 简洁 明了 。 

还 有 另 一 种 设计 考虑 ， 即 你 确实 希望 允许 客户 端 程序 员 改 变 first 或 second 所 引用 的 对 象 。 然 
而 ， 采 用 以 上 的 形式 无 疑 是 更 安全 的 做 法 ， 这 样 的 话 ， 如 果 程序 员 想 要 使 用 具有 不 同 元 素 的 元 
组 ， 就 强制 要 求 他 们 另外 创建 一 个 新 的 TwoTuple 对 象 。 

我 们 可 以 利用 继承 机 制 实现 长 度 更 长 的 元 组 。 从 下 面 的 例子 中 可 以 看 到 ， 增 加 类 型 参数 是 
件 很 简单 的 事情 : 


//: net/mindview/util/ThreeTuple. java 
Package net.mindview.util; 


public class ThreeTuple<A,B,C> extends TwoTuple<A,B> { 
public final C third; 
public ThreeTuple(A a, B b, Cc) { 
super(a, b); 
third = c; 


} 
public String toString() { 
return "(" + first +", " + second +", " + third +")"; 
} 
} Mi~ 
//: net/mindview/util/FourTuple. java 
package net.mindview.util; 


public class FourTuple<A,B,C,D> extends ThreeTuple<A,B,C> { 
public final D fourth; 
public FourTuple(A a, B b, C c, D d) { 
super(a, b, c); 
fourth = d; 


} 
Public String toString() { 
return "(" + first +", "+ second +", "+ 
third +", " + fourth + ")"; 


} 
} Mi~ 


//: net/mindview/util/FiveTuple. java 
package net.mindview.util; 


public class FiveTuple<A,B.C,D,E> 
extends FourTuple<a,B,C,D> { 
public final E fifth; 
public FiveTuple(A a, B b, Cc, Dd, Ee) { 
super(a, b, c. d); 
fifth = e; 


} 
Public String toString() { 
return "(" + first +", "+ second +", "+ 
third +", "+ fourth +", "+ fifth + ")"; 


} 
} Ii~ 
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为 了 使 用 元 组 ， 你 只 需 定义 一 个 长 度 适 合 的 元 组 ， 将 其 作为 方法 的 返回 值 ， 然 后 在 return 语 
句 中 创建 该 元 组 ， 并 返回 即 可 。 


//: generics/TupleTest. java 
import net.mindview.util.*; 


class Amphibian {} 
class Vehicle {} 


public class TupleTest { 
static TwoTuple<String,Integer> fO { 
// Autoboxing converts the int to Integer: 
return new TwoTuple<String, Integer>("hi", 47); 


} y 
static ThreeTuple<Amphibian, String, Integer> g() { 
return new ThreeTuple<Amphibian, String, Integer>( 
new Amphibian(), “hi", 47); 
} 
static 
FourTuple<Vehicle,Amphibian, String, Integer> hO { 
return 
new FourTuple<Vehicle, Amphibian, String, Integer>( 
new Vehicle(), new Amphibian(), "hi", 47); 
} 
static 
FiveTuple<Vehicle, Amphibian, String, Integer ,Double> k() { 
return new 
FiveTuple<Vehicle, Amphibian, String, Integer ,Double>( 
new Vehicle(), new Amphibian(), "hi", 47, 11.1); 
) 
public static void main(String() args) { 
‘TwoTuple<String.Integer> ttsi = f(); 
System. out.printLn(ttsi); 
// ttsi.first = "there"; // Compile error: final 
System.out.printin(g()); 
System.out.printin(h()); 
System. out.println(k()); 


} » Output: (80% match) 

(hi, 47) 

(Amphibian@1f6a7b9, hi, 47) 

(Vehicle@35ce36, Amphibian@757aef, hi, 47) 

by hag ade Amphibian@la46e36, hi, 47, 11.1) 

由 于 有 了 泛 型 ， 你 可 以 很 容易 地 创建 元 组 ， 令 其 返回 一 组 任意 类 型 的 对 象 。 而 你 所 要 做 的 ， 
只 是 编写 表达 式 而 已 。 

通过 ttsifirst = "there" 语 句 的 错误 ， 我 们 可 以 看 出 ，final 声 明确 实 能 够 保护 public 元 素 ， 在 
对 象 被 构造 出 来 之 后 ， 声 明 为 final 的 元 素 便 不 能 被 再 赋予 其 他 值 了 。 

在 上 面 的 程序 中 ，new 表 达 式 确实 有 点 罗 嗪 。 本 章 稍 后 会 介绍 ， 如 何 利用 泛 型 方法 简化 这 样 
的 表达 式 。 

练习 3: (1) 使 用 泛 型 编写 一 个 SixTuple 类 ， 并 测试 它 。 

练习 4: (3)“ 泛 型 化 ”innerclasses/Sequence.java 类 。 
15.2.2 一 个 堆栈 类 

接 下 来 我 们 看 一 个 稍微 复杂 一 点 的 例子 : 传统 的 下 推 堆栈 。 在 第 11 章 中 ， 我 们 看 到 ， 这 个 堆 
栈 是 作为 net.mindview.util.Stack 类 ， 用 一 个 LinkedList 实 现 的 。 在 那个 例子 中 ，LinkedList 本 身 
已 经 具备 了 创建 堆栈 所 必需 的 方法 ， 而 Stack 可 以 通过 两 个 泛 型 的 类 Stack<T> 和 LinkedList<T> 
的 组 合 来 创建 。 在 那个 示例 中 ， 我 们 可 以 看 出 ， 泛 型 类 型 也 就 是 另 一 种 类 型 罢了 (WERNA 
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看 到 一 些 例外 的 情况 )。 
现在 我 们 不 用 LinkedList， 来 实现 自己 的 内 部 链 式 存储 机 制 。 


//: generics/LinkedStack. java 
// A stack implemented with an internal linked structure. 


public class LinkedStack<T> { 
private static class Node<U> { 

U item; 

Node<U> next; 

Node() { item = null; next = null; } 

Node(U item, Node<U> next) { 
this.item = item; 
this.next = next; 


} 

boolean end() { return item == null && next == null; } 
} 
private Node<T> top = new Node<T>(); // End sentinel 
public void push(T item) { 

top = new Node<T>(item, top); 


} 
public T pop() { 
T result = top.item; 
if(!top.end()) 
top = top.next; 
return result; 
} 
public static void main(String{] args) { 
LinkedStack<String> lss = new LinkedStack<String>(); 
for(String s : "Phasers on stun!".split(" ")) 
Uss.push(s); 


String s; 
while((s = 1ss.pop()) != nuti) 
System.out.printtn(s); 


} 
} /* Output: 
stun! 
on 
Phasers 
Wim 


内 部 类 Node 也 是 一 个 泛 型 ， 它 拥有 自己 的 类 型 参数 。 

这 个 例子 使 用 了 一 个 末端 哨兵 (end sentinel) 来 判断 堆栈 何 时 为 空 。 这 个 末端 哨兵 是 在 构 
造 LinkedStack 时 创建 的 。 然后， 每 调用 一 次 push0 方 法 ， 就 会 创建 一 个 Node<T> 对 象 ， 并 将 其 
链接 到 前 一 个 Node<T> 对 象 。 当 你 调用 pop0 方 法 时 ， 总 是 返回 top.item， 然 后 丢弃 当前 top 所 指 

` 的 Node<T>， 并 将 top 转 移 到 下 一 个 Node<T>， 除 非 你 已 经 碰 到 了 末端 哨兵 ， 这 时 候 就 不 再 移动 

top 了 。 如 果 已 经 到 了 末端 ， 客 户 端 程序 还 继续 调用 pop0 方 法 ， 它 只 能 得 到 null， 说 明 堆栈 已 经 
aT. 

练习 5: (2) 移 除 Node 类 上 的 类 型 参数 ， 并 修改 LinkedStack.java 的 代码 ， 证 明 内 部 类 可 以 访 
问 其 外 部 类 的 类 型 参数 。 
15.2.3 RandomList 

作为 容器 的 另 一 个 例子 ， 假 设 我 们 需要 一 个 持 有 特定 类 型 对 象 的 列表 ， 每 次 调用 其 上 的 
select0 方 法 时 ， 它 可 以 随机 地 选取 一 个 元 素 。 如 果 我 们 希望 以 此 构建 一 个 可 以 应 用 于 各 种 类 型 
的 对 象 的 工具 ， 就 需要 使 用 泛 型 : 


| //: generics/RandomList.java 
import java.util.*; 


public class RandomList<T> { 
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private ArrayList<T> storage = new ArrayList<T>(); 
private Random rand = new Random(47); 
public void add(T item) { storage.add(item); } 
public T select() { 

return storage. get(rand.nextInt(storage.size())); 


public static void main(String[] args) { 
RandomList<String> rs = new RandomList<String>(); 
for(String s: ("The quick brown fox jumped over * + 
“the lazy brown dog").split(” ")) 
rs.add(s); 
for(int 1 = @; 1 < 11; i++) 
System.out.print(rs.select() + ™ "); 


} 
} /* Output: 
brown over fox quick quick dog brown The brown lazy brown 
AAA 


练习 6: (1) 使 用 RandomList 来 处 理 两 种 额外 的 不 同类 型 的 元 素 ， 要 区 别 于 main0 中 已 经 用 
过 的 类 型 。 


. 15.3 泛 型 接口 


泛 型 也 可 以 应 用 于 接口 。 例 如 生成 器 (generator)， 这 是 一 种 专门 负责 创建 对 象 的 类 。 实 际 
上 ， 这 是 工厂 方法 设计 模式 的 一 种 应 用 。 不 过 ， 当 使 用 生成 器 创建 新 的 对 象 时 ， 它 不 需要 任何 
参数 ， 而 工厂 方法 一 般 需 要 参数 。 也 就 是 说 ， 生 成 器 无 需 额外 的 信息 就 知道 如 何 创建 新 对 象 。 

一 般 而 言 ， 一 个 生成 器 只 定义 一 个 方法 ， 该 方法 用 以 产生 新 的 对 象 。 在 这 里 ， 就 是 next0 方 
法 。 我 将 它 收录 在 我 的 标准 工具 类 库 中 : 


//: net/mindview/util/Generator . java 
// A generic interface. 

package net.mindview.util; 

public interface Generator<T> { T next(); } ///:~ 


方法 next0 的 返回 类 型 是 参数 化 的 T。 正 如 你 所 见 到 的 ， 接 口 使 用 泛 型 与 类 使 用 泛 型 没什么 
区 别 。 
为 了 演示 如 何 实现 Generator 接 口 ， 我 们 还 需要 一 些 别 的 类 。 例 如 ，Coffee 类 层次 结构 如 下 ; 


//: generics/coffee/Coffee. java 
package generics.coffee; 


public class Coffee { 
private static long counter = 6; 
private final long id = counter++; 
public String toString() { 
return getClass().getSimpleName() + * + id; 
} 
} Mi~ 


//: generics/coffee/Latte. java 


package generics.coffee; 
public class Latte extends Coffee {} ///:~ 


//: generics/coffee/Mocha.java 
package generics.coffee; 
public class Mocha extends Coffee {} ///:~ 


//: generics/coffee/Cappuccino. java 
package generics. coffee; 
public class Cappuccino extends Coffee {} ///:~ 


//: generics/coffee/Americano. java 
package generics .coffee; 
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Coffeext $% 


public class Americano extends Coffee {} ///:~ 


//: generics/coffee/Breve. java 
package generics.coffee; 
public class Breve extends Coffee {} ///:~ 


现在 ， 我 们 可 以 编写 一 个 类 ， 实 现 Generator<Coffee> 接 口 ， 它 能 够 随机 生成 不 同类 型 的 





//: generics/coffee/Cof feeGenerator. java 
// Generate different types of Coffee: 
package generics.coffee; 
import java.util.*; 
import net.mindview.util. 





public class CoffeeGenerator 
implements Generator<Coffee>, Iterable<Coffee> { 
private Class[] types = { Latte.class, Mocha.class, 
Cappuccino.class, Americano.class, Breve.class, }; 
private static Random rand = new Random(47); 
public CoffeeGenerator() {} 
// For 1teratio 
private int size = 0; 
public CoffeeGenerator(int sz) { size = sz; } 
public Coffee next() { 
try { 
return (Coffee) 
types [rand.nextInt (types. Length) ] .newInstance() ; 
// Report programmer errors at run time: 
} catch(Exception e) { 
throw new RuntimeException(e) : 
) 
} 


class Coffeelterator implements Iterator<Coffee> { 
‘int count = size; 
public boolean hasNext() { return count > @; } 
public Coffee next() { 
count-~; 
return CoffeeGenerator.this.next(); 
} 
public void remove() { // Not implemented 
throw new UnsupportedOperat ionExcept ion(); 
} 
}; 
public Iterator<Coffee> iterator() { 
return new Coffeelterator(); 
} 
public static void main(String{) args) { 
CoffeeGenerator gen = new CoffeeGenerator(); 
for(int 1 = 0; 1 < 5; i++) 
System. out.printin(gen.next()); 
for (Coffee c : new CoffeeGenerator(5)) 
System.out.printtn(c); 





} 
} /* Output: 
Americano © 
Latte 1 
Americano 2 
Mocha 3 
Mocha 4 
Breve 5 
Americano 6 
Latte 7 
Cappuccino 8 
Cappuccino 9 
“Ii~ 
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参数 化 的 Generator 接 口 确保 next0 的 返回 值 是 参数 的 类 型 。CoffeeGenerator 同 时 还 实现 了 
Iterable 接 口 ， 所 以 它 可 以 在 循环 语句 中 使 用 。 不 过 ， 它 还 需要 一 个 “末端 哨兵 ”来 判断 何 时 停 
止 ， 这 正 是 第 二 个 构造 器 的 功能 。 

下 面 的 类 是 Generator<T> 接 口 的 另 一 个 实现 ， 它 负责 生成 Fibonacci 数 列 ; 


(629) //: generics/Fibonacci. java 











// Generate a Fibonacci sequence. 
import net.mindview.util.*; 


public class Fibonacci implements Generator<Integer> { 
private int count = @; 
public Integer next() { return fib(count++); } 
private int fib(int n) { 
if(n < 2) return 1; 
return fib(n-2) + fib(n-1); 


} 
public static void main(String(] args) { 
Fibonacci gen = new Fibonacci (); 
for(int 1 = 0; 1 < 18; i++) 
System.out.print(gen.next() +" "); 


} hs Output: 

1/1/2358 13 21 34 55 89 144 233 377 610 987 1597 2584 

虽然 我 们 在 Fibonacei 类 的 里 里 外 外 使 用 的 都 是 int 类 型 ， 但 是 其 类 型 参数 却 是 Integer。 这 个 
例子 引出 了 Java 泛 型 的 一 个 局 限 性 ; 基本 类 型 无 法 作为 类 型 参数 。 不 过 ，Java SE5 具 备 了 自动 打 
包 和 自动 拆 包 的 功能 ， 可 以 很 方便 地 在 基本 类 型 和 其 相应 的 包装 器 类 型 之 间 进 行 转换 。 通 过 这 
个 例子 中 Fibonacei 类 对 int 的 使 用 ， 我 们 已 经 看 到 了 这 种 效果 。 

如 果 还 想 更 进一步 ， 编 写 一 个 实现 了 Iterable 的 Fibonacci 生 成 器 。 我 们 的 一 个 选择 是 重 写 这 
个 类 ， 令 其 实现 Iterable 接 口 。 不 过 ， 你 并 不 是 总 能 拥有 源 代码 的 控制 权 ， 并 且 ， 除 非 必须 这 么 
做 ， 否 则 ， 我 们 也 不 愿意 重 写 一 个 类 。 而 且 我 们 还 有 另 一 种 选择 ， 就 是 创建 一 个 适配器 (adapter) 
来 实现 所 需 的 接口 ， 我 们 在 前 面 介 绍 过 这 个 设计 模式 。 

有 多 种 方法 可 以 实现 适配器 。 例 如 ， 可 以 通过 继承 来 创建 适配器 类 ， 


//: generics/IterableFibonacci. java 
// Adapt the Fibonacci class to make it Iterable. 
import java.util.*; 


public class IterableFibonacci 
extends Fibonacci implements Iterable<Integer> { 
private int n; 
public IterableFibonacci(int count) { n = count; } 
630, public Iterator<Integer> iterator() { 
return new Iterator<Integer>() { 
public boolean hasNext() { return n > 6; } 
Public Integer next() { 
pas 














return IterableFibonacci.this.next(); 


public void remove() { // Not implemented 
throw new UnsupportedOperationException(); 
} 
}; 
public static void main(String) args) { 
for(int i : new IterableFibonacci(18)) 
System.out.print(i + * "); 


} 
} /* Output: 
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11235 8 13 21 34 55 89 144 233 377 619 987 1597 2584 
When 


如 果 要 在 循环 语句 中 使 用 IterableFibonacci， 必 须 向 IterableFibonacci 的 构造 器 提供 一 个 边 
界 值 ， 然 后 hasNext0 方 法 才能 知道 何 时 应 该 返回 false。 

练习 7: (2) 使 用 组 合 代替 继承 ， 适 配 Fibonacci 使 其 成 为 Iterable。 

练习 8: (2) 模仿 Coffee 示 例 的 样子 ， 根 据 你 喜爱 的 电影 人 物 ， 创 建 一 个 StoryCharacters 的 
类 层次 结构 ， 将 它们 划分 为 GoodGuys 和 BadGuys。 再 按照 CoffeeGenerator 的 形式 ， 编 写 一 个 
StoryCharacters 的 生成 器 。 


15.4 泛 型 方法 


到 目前 为 止 ， 我 们 看 到 的 泛 型 ， 都 是 应 用 于 整个 类 上 。 但 同样 可 以 在 类 中 包含 参数 化 方法 ， 
而 这 个 方法 所 在 的 类 可 以 是 泛 型 类 ， 也 可 以 不 是 泛 型 类 。 也 就 是 说 ， 是 否 拥有 泛 型 方法 ， 与 其 
所 在 的 类 是 否 是 泛 型 没有 关系 。 

泛 型 方法 使 得 该 方法 能 够 独立 于 类 而 产生 变化 。 以 下 是 一 个 基本 的 指导 原则 无 论 何 时 ， 
只 要 你 能 做 到 ， 你 就 应 该 尽量 使 用 泛 型 方法 。 也 就 是 说 ， 如 果 使 用 泛 型 方法 可 以 取代 将 整个 类 
泛 型 化 ， 那 么 就 应 该 只 使 用 泛 型 方法 ， 因 为 它 可 以 使 事情 更 清楚 明白 。 另 外 ， 对 于 一 个 statie 的 
方法 而 言 ， 无 法 访问 泛 型 类 的 类 型 参数 ， 所 以 ， 如 果 static 方 法 需要 使 用 泛 型 能 力 ， 就 必须 使 其 
成 为 泛 型 方法 。 

要 定义 泛 型 方法 ， 只 需 将 泛 型 参数 列表 置 于 返回 值 之 前 ， 就 像 下 面 这 样 : 

//: generics/GenericMethods. java 

public class GenericMethods { 


public <T> void f(T x) { 
System.out.printin(x.getClass().getName()) ; 


$ 
public static void main(String[] args) { 
GenericHethods gm = new GenericMethods(); 
gmt"); 
gm. t (1); 
gm. f (1.0); 
gm.f(1.0F) ; 
Bm. f('C'); 
gm. f(gm); 


} /* Output: 
java.lang.String 
java. tang. Integer 
java. lang. Double 
java. lang. Float 
java. lang.Character 
Gener icMethods 
Whim 


GenericMethods 并 不 是 参数 化 的 ， 尽 管 这 个 类 和 其 内 部 的 方法 可 以 被 同时 参数 化 ， 但 是 在 
这 个 例子 中 ， 只 有 方法 f0 拥 有 类 型 参数 。 这 是 由 该 方法 的 返回 类 型 前 面 的 类 型 参数 列表 指明 的 。 

注意 ， 当 使 用 泛 型 类 时 ， 必 须 在 创建 对 象 的 时 候 指定 类 型 参数 的 值 ， 而 使 用 泛 型 方法 的 时 
候 ， 通 常 不 必 指 明 参 数 类 型 ， 因 为 编译 器 会 为 我 们 找 出 具体 的 类 型 。 这 称 为 类 型 参数 推断 (type 
argument inference)。 因 此 ， 我 们 可 以 像 调用 普通 方法 一 样 调用 f0， 而 且 就 好 像 是 f0 被 无 限 次 地 
重 载 过 。 它 甚至 可 以 接受 GenericMethods 作 为 其 类 型 参数 。 

如 果 调 用 f0 时 传人 基本 类 型 ， 自 动 打 包机 制 就 会 介入 其 中 ， 将 基本 类 型 的 值 包装 为 对 应 的 
对 象 。 事 实 上 ， 泛 型 方法 与 自动 打包 各 免 了 许多 以 前 我 们 不 得 不 自己 编写 出 来 的 代码 。 
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练习 9: (1) 修改 GenericMethodsjava 类 ， 使 ?0 可 以 接受 三 个 类 型 各 不 相同 的 参数 。 
练习 10: (1) 修改 前 一 个 练习 ， 将 方法 f0 的 其 中 一 个 参数 修改 为 非 参数 化 的 类 型 


15.4.1 杠杆 利用 类 型 参数 推断 
人 们 对 泛 型 有 一 个 抱怨 ， 使 用 泛 型 有 时 候 需要 向 程序 中 加 入 更 多 的 代码 。 考 虑 第 11 章 中 的 
holding/MapOfListjava 类 ， 如 果 要 创建 一 个 持 有 List 的 Map， 就 要 像 下 面 这 样 ; 


Map<Person, List<? extends Pet>> petPeople = 
new HashMap<Person, List<? extends Pet>>(); 


(本 章 稍 后 会 介绍 表达 式 中 问号 与 extends 的 用 法 。) 看 到 了 吧 ， 你 在 重复 自己 做 过 的 事情 ， 
编译 器 本 来 应 该 能 够 从 泛 型 参数 列表 中 的 一 个 参数 推断 出 另 一 个 参数 。 唉 ， 可 惜 的 是 ， 编 译 器 
暂时 还 做 不 到 。 然 而 ， 在 泛 型 方法 中 ， 类 型 参数 推断 可 以 为 我 们 简化 一 部 分 工作 。 例 如 ， 我 们 
可 以 编写 一 个 工具 类 ， 它 包含 各 种 各 样 的 static 方 法 ， 专 门 用 来 创建 各 种 常用 的 容器 对 象 : 


/1: net/mindview/util/New. java 
// Utilities to simplify generic container creation 
// by using type argument inference. 

package net.mindview.util; 

import java.util.*; 


public class New { 

public static <K,V> Map<K,V> map() { 
return new HashMap<k,V>(); 

) 

public static <T> List<T> List() { 
return new ArrayList<T>(); 

} 

public static <T> LinkedList<T> IList() { 
return new LinkedList<T>(); 


} 

public static <T> Set<T> set() { 
return new HashSet<T>(); 

} 

public static <T> QueuecT> queue() { 
return new LinkedList<T>(); 


$ 

// Examples: 

public static void main(String[] args) { 
Map<String, List<String>> sls = New.map(); 
List<String> Is = New. listQ); 
LinkedList<String> lls = New.1List(); 
Set<String> ss = New.set(); 
Queue<String> qs = New.queue(); 


} 
dM 
main() 方 法 演示 了 如 何 使 用 这 个 工具 类 ， 类 型 参数 推断 避免 了 重复 的 泛 型 参数 列表 。 它 同 
样 可 以 应 用 于 holding/MapOfListjava: 


//: generics/Simplerpets.java 
import typeinfo.pets.*; 
import java.util.*; 

import net.mindview.util.*; 


public class SimplerPets { 
public static void main(String[] args) { 
Map<Person, List<? extends Pet>> petPeople = New.map(); 
// Rest of the code is the same... 
YUH 


对 于 类 型 参数 推断 而 言 ， 这 是 一 个 有 趣 的 例子 。 不 过 ， 很 难说 它 为 我 们 带 来 了 多 少 好 处 。 
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如 果 某 人 阅读 以 上 代码 ， 他 必须 分 析 理解 工具 类 New， 以 及 New 所 隐 含 的 功能 。 而 这 似乎 与 不 
使 用 New 时 (具有 重复 的 类 型 参数 列表 的 定义 ) 的 工作 效率 差不多 。 这 真 够 讽刺 的 ， 要 知道 ， 
我 们 引入 New 工 具 类 的 目的 ， 正 是 为 了 使 代码 简单 易 读 。 不 过 ， 如 果 标准 Java 类 库 要 是 能 添加 类 
似 Newjava 这 样 的 工具 类 的 话 ， 我 们 还 是 应 该 使 用 这 样 的 工具 类 。 

类 型 推断 只 对 赋值 操作 有 效 ， 其 他 时 候 并 不 起 作用 。 如 果 你 将 一 个 泛 型 方法 调用 的 结果 
(例如 New.mapO) 作为 参数 ， 传 递 给 另 一 个 方法 ， 这 时 编译 器 并 不 会 执行 类 型 推断 。 在 这 种 情 
况 下 ， 编 译 器 认为 ， 调 用 泛 型 方法 后 ， 其 返回 值 被 赋 给 一 个 Object 类 型 的 变量 。 下 面 的 例子 证 
明了 这 一 点 : 


//: generics/LimitsOfInference. java 
import typeinfo.pets.*; 
import java.util.*; 
public class LimitsOf Inference { 
static void 
f(MapsPerson, List<? extends Pet>> petPeople) {} 
public static void main(String{) args) { 
// £(New.map()); // Does not compile 


} 
Uhm 


练习 11，(1) 创建 自己 的 若干 个 类 来 测试 Newjava， 并 确保 New 可 以 正确 地 与 它们 一 起 工作 。 

显 式 的 类 型 说 明 

在 泛 型 方法 中 ， 可 以 显 式 地 指明 类 型 ， 不 过 这 种 语法 很 少 使 用 。 要 显 式 地 指明 类 型 ， 必 须 
在 点 操作 符 与 方法 名 之 间 插 入 尖 括 号 ， 然 后 把 类 型 置 于 尖 括 号 内 。 如 果 是 在 定义 该 方法 的 类 的 
内 部 ， 必 须 在 点 操作 符 之 前 使 用 this 关 键 字 ， 如 果 是 使 用 statie 的 方法 ， 必 须 在 点 操作 符 之 前 加 
上 类 名 。 使 用 这 种 语法 ， 可 以 解决 LimitsOfInference.java 中 的 问题 : 


//: generics/ExplicitTypeSpecification. java 
import typeinfo.pets.*; 

import java.util.*; 

import net.mindview.util.*: 


public class ExplicitTypeSpecification { 
static void f(Map<Person, List<Pet>> petPeople) {} 
public static void main(String[] args) { 
f (New. <Person, List<Pet>>map()): 


} 
} Mi~ 


当然 ， 这 种 语法 抵消 了 New 类 为 我 们 带 来 的 好 处 〈 即 省 去 了 大 量 的 类 型 说 明 ) ， 不 过 ， 只 有 
在 编写 非 赋值 语 名 时， 我们 才 需要 这 样 的 额外 说 明 。 
练习 12: (1) 使 用 显 式 的 类 型 说 明 来 重复 前 一 个 练习 。 


15.4.2 可 变 参 数 与 泛 型 方法 
泛 型 方法 与 可 变 参 数列 表 能 够 很 好 地 共存 : 


//: generics/GenericVarargs.java 
import. java.util.*; 


public class GenericVarargs { 
Public static <T> List<T> makeList(T... args) { 
ListsT> result = new Arraylist<T>(); 
for(T item : args) 
result.add(item) ; 
return result; 


} 
public static void main(String[] args) { 
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List<String> Ls = makeList("A"); 
System.out.printin(1s) ; 
1s = makeList("A", "B 
System.out .printin(1s); 
Us = makeList ("ABCDEFFHIJKLMNOPQRSTUVWXYZ".split("")); 
System. out.printIn(1s); 





"C"); 


$ 
} /* Output: 


FL FH. 0, J, KL, M,N, 0, P,Q, RS, 
2) 


makeList0 方 法 展示 了 与 标准 类 库 中 java.util.Arrays.asList0 方 法 相同 的 功能 。 


15.4.3 用 于 Generator 的 泛 型 方法 
利用 生成 器 ， 我 们 可 以 很 方便 地 填充 一 个 Collection， 而 泛 型 化 这 种 操作 是 具有 实际 意义 的 ; 


//: generics/Generators. java 
// A utility to use with Generators. 
import generics. coffee.*; 

import java.util.*; 

import net.mindview.util.*; 


public class Generators { 
public static <T> Collection<T> 
fill (Collection<T> coll, Generator<T> gen, int n) { 
(636) for(int i = @; i < n; 1++) 
coll.add(gen.next()); 
return coll; 
} 
public static void main(String{] args) { 
Collection<Coffee> coffee = fill( 
new Arraylist<Coffee>(), new CoffeeGenerator(), 4); 
for (Coffee c : coffee) 
System.out.printin(c); 
i Collection<Integer> fnumbers = fill( 
new ArrayList<Integer>(), new Fibonacci(), 12); 
for(int 1 : fnumbers) 
System.out.print(i +", "); 
} 


} /* Output: 
Americano © 

Latte 1 

‘Americano 2 

Mocha 3 

1, 1, 2, 3, 5, 8, 13, 21, 34, 55, 89, 144, 
li~ 


请 注意 ，fill0 方 法 是 如 何 透明 地 应 用 于 Coffee 和 Integer 的 容器 和 生成 器 。 
练习 13:(4) 重 载 {il0 方 法， 使 其 参数 与 返回 值 的 类 型 为 Collection 的 导出 类 ;List、Queue 
和 Set。 通 过 这 种 方式 ， 我 们 就 不 会 丢失 容器 的 类 型 。 能 够 在 重 载 时 区 分 List 和 LinkedList 吗 ? 
15.4.4 一 个 通用 的 Generator 
下 面 的 程序 可 以 为 任何 类 构造 一 个 Generater， 只 要 该 类 具有 默认 的 构造 器 。 为 了 减少 类 型 
| 声明 ， 它 提供 了 一 个 泛 型 方法 ， 用 以 生成 BasicGenerator: 


//: net/mindview/util/BasicGenerator.java 

// Automatically create a Generator, given a class 
// with a default (no-arg) constructor. 

package net.mindview.util; 




















public class BasicGenerator<T> implements Generator<T> { 
private Class<T> type; 
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public BasicGenerator(Class<T> type){ this.type = type; } 
public T next() { 
try { 
// Assumes type is a public class: 
return type.newInstance(); 
} catch(Exception e) { 
throw new RuntimeException(e); 


} 
// Produce a Default generator given a type token: 
public static <T> Generator<T> create(Class<T> type) { 
return new BasicGenerator<T> (type): 
} 
} :~ 
这 个 类 提供 了 一 个 基本 实现 ， 用 以 生成 某 个 类 的 对 象 。 这 个 类 必需 具备 两 个 特点 ，(1) È 


必须 声明 为 publie。 (因为 BasicGenerator 与 要 处 理 的 类 在 不 同 的 包 中 ， 所 以 该 类 必须 声明 为 


public， 并 且 不 只 具有 包 内 访问 权限 。.) (2) 它 必须 具备 默认 的 构造 器 (无 参数 的 构造 器 )。 要 创 
建 这 样 的 BasicGenerator 对 象 ， 只 需 调用 create0) 方 法 ， 并 传人 想 要 生成 的 类 型 。 泛 型 化 的 
create() 方 法 允许 执行 BasicGenerator.create(MyType.class)， 而 不 必 执 行 麻烦 的 new Basic- 
Generator<MyType>(MyType.class) , 

例如 ， 下 面 是 一 个 具有 默认 构造 器 的 简单 的 类 : 


1/: generics/CountedObject. java 


public class CountedObject { 
private static long counter = 0; 
private final long id = counter++; 
public long id() { return id; } 
public String toString() { return "CountedObject " + 1d;} 
bh 


CountedObject 类 能 够 记录 下 它 创建 了 多 少 个 CountedObject 实 例 ， 并 通过 toString0 方 法 告 
诉 我 们 其 编号 。 
使 用 BasicGenerator， 你 可 以 很 容易 地 为 CountedObject 创 建 一 个 Generator: 


//: generics/BasicGeneratorDemo. java 
import net.mindview.util.*; 


public class BasicGeneratorDemo { 
public static void main(String[] args) { 
Generator<CountedObject> gen = 
BasicGenerator .create(CountedObject.class); 
for(int 1 = 0; 1 < 5; 1++) 
System. out.printin(gen.next()); 


} 
} /* Output: 
CountedObject @ 
CountedObject 1 
CountedObject 2 A 
CountedObject 3 
CountedObject 4 
i~ 


可 以 看 到 ， 使 用 泛 型 方法 创建 Generator 对 象 ， 大 大 减少 了 我 们 要 编写 的 代码 。Java 泛 型 要 
求 传人 Class 对 象 ， 以 便 也 可 以 在 create0 方 法 中 用 它 进行 类 型 推断 。 

练习 14: (1) 修改 BasicGeneratorDemo.java 类 ， 使 其 显 式 地 构造 Generator (也 就 是 不 使 用 
create0 方 法 ， 而 是 使 用 显 式 的 构造 器 )。 
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15.4.5 简化 元 组 的 使 用 
有 了 类 型 参数 推断 ， 再 加 上 static 方 法 ， 我 们 可 以 重新 编写 之 前 看 到 的 元 组 工具 ， 使 其 成 为 
更 通用 的 工具 类 库 。 在 这 个 类 中 ， 我 们 通过 重 载 static 方 法 创建 元 组 : 


/1:_net/mindview/util/Tuple. java 
// Tuple library using type argument inference. 
package net.mindview.util; 


public class Tuple { 
public static <A,B> TwoTuple<A,B> tuple(A a, B b) { 
return new TwoTuple<A,B>(a, b); 
} 
public static <A,B,C> ThreeTuple<A,B,C> 
tuple(A a, Bb, Cc) { 
return new ThreeTuple<A,B,C>(a, b, c); 
} 
public static <A,B,C,D> FourTuple<A,B.C,D> 
tuple(A a, Bb, C c, Dd) { 
return new FourTuple<A,8,C,0>(a, b, c, d); 
} 
public static <A,B,C,D,E> 
FiveTuple<a,8,C,0,E> tuple(A a, B b, Cc, Dd, Ee) { 
return new FiveTuple<A,8,C,D,£>(a, b, c, d, e); 
} 
} Mi~ 


下 面 是 修改 后 的 TupleTestjava， 用 来 测试 Tuplejava: 


//: generics/TupleTest2. java 
‘import net.mindview.util.*; 
import static net.mindview.util.Tuple.*; 











public class TupleTest2 { 
static TwoTuple<String,Integer> f() { 
return tuple("hi", 47); 
} 
static TwoTuple f2() { return tuple("hi", 47); } 
static ThreeTuple<Anphibian, String, Integer> g() { 
return tuple(new Amphibian(), “hi”, 47); 
k 
static 
FourTuple<Vehicle,Amphibian,String,Integer> h() { 
return tuple(new Vehicle(), new Amphibian(), “hi”, 47); 
} 
static 
FiveTuple<Vehicle, Amphibian, String, Integer ,Double> k() { 
return tuple(new Vehicle(), new Amphibian(), 
“hie, aaisa 
} 


public static void main(String[] args) { 
TwoTuple<String,Integer> ttsi = f(): 
System.out.printin(ttsi); 
System. out.printin(f2()); 
System. out.printin(g()): 
System. out.printin(h()): 
System.out.println(k()): 
} 
} /* Output: (80% match) 
(hi, 47) 
(hi, 47) 
(Amphibian@7d772e, hi, 47) 
(Vehicle@757aef, Amphibian@d9f9c3, hi, 47) 
(Vehicle@1a46e30, Amphibian@3e25a5, hi, 47, 11.1) 
Wh 


注意 ， 方 法 (0 返回 一 个 参数 化 的 TwoTuple 对 象 ， 而 人 220 返回 的 是 非 参 数 化 的 TwoTuple 对 象 。 
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在 这 个 例子 中 ， 编 译 器 并 没有 关于 人 0 的 警告 信息 ， 因 为 我 们 并 没有 将 其 返回 值 作为 参数 化 对 象 
使 用 。 在 某 种 意义 上 ， 它 被 “向 上 转型 ”为 一 个 非 参数 化 的 TwoTuple。 然 而 ， 如 果 试 图 将 20 的 
返回 值 转型 为 参数 化 的 TwoTuple， 编 译 器 就 会 发 出 警告 。 

练习 15，(1) 验证 前 面 的 陈述 是 否 属实 。 

练习 16， (2) 为 Tuplejava 添 加 一 个 SixTuple， 并 在 TupleTest2.java 中 进行 测试 。 
15.4.6 一 个 Set 实 用 工具 

作为 泛 型 方法 的 另 一 个 示例 ， 我 们 看 看 如 何 用 Set 来 表达 数学 中 的 关系 式 。 通 过 使 用 泛 型 方 
法 ， 可 以 很 方便 地 做 到 这 一 点 ， 而 且 可 以 应 用 于 多 种 类 型 : 


/1/: net/mindview/util/Sets. java 
package net .mindview.util; 
import java.util.*; 


public class Sets { 

public static <T> Set<T> unfon(Set<T> a, Set<T> b) { 
Set<T> result = new HashSet<T>(a); 
result. addAll(b); 
return result; 

} 

public static <T> 

Set<T> intersection(Set<T> a, Set<T> b) { 
Set<T> result = new HashSet<T>(a); 
result.retainAll(b); 
return result; 


// Subtract subset from superset: 

public static <T> Set<T> 

difference(Set<T> superset, Set<T> subset) { 
Set<T> result = new HashSet<T>(superset); 
result. removeAll (subset) ; 
return result; 


} 

// Reflexive--everything not in the intersection: 

public static <T> Set<T> complement(Set<T> a, Set<T> b) { 
return difference(union(a, b), intersection(a, b)); 


} 
} Mi~ 


在 前 三 个 方法 中 ， 都 将 第 一 个 参数 Set 复 制 了 一 份 ， 将 Set 中 的 所 有 引用 都 存 人 一 个 新 的 
HashSet 对 象 中 ， 因 此 ， 我 们 并 未 直接 修改 参数 中 的 Set。 返 回 的 值 是 一 个 全 新 的 Set 对 象 。 

这 四 个 方法 表达 了 如 下 的 数学 集合 操作 :union0 返 回 一 个 Set， 它 将 两 个 参数 合并 在 一 起 ， 
intersection0 返 回 的 Set 只 包含 两 个 参数 共有 的 部 分 ，difference( 方 法 从 superset 中 移 除 subset 包 
含 的 元 素 ，complement0 返 回 的 Set 包 含 除了 交集 之 外 的 所 有 元 素 。 下 面 提供 了 一 个 enum， 它 包 
含 各 种 水 彩 画 的 颜色 。 我 们 将 用 它 来 演示 以 上 这 些 方法 的 功能 和 效果 。 


//: generics/watercolors/Watercolors.java 
package generics.watercolors; 


public enum Watercolors { 
ZINC, LEMON_YELLOW, MEDIUM_YELLOW, DEEP_YELLOW, ORANGE, 
BRILLIANT_RED, CRIMSON, MAGENTA, ROSE_MADDER, VIOLET, 
CERULEAN_BLUE_HUE, PHTHALO_BLUE, ULTRAMARINE, 
COBALT_BLUE_HUE, PERMANENT_GREEN, VIRIDIAN_HUE, 
SAP_GREEN, YELLOW_OCHRE, BURNT_SIENNA, RAW_UMBER, 
BURNT_UMBER, PAYNES GRAY, IVORY_BLACK š 
y= 


为 了 方便 起 见 ( 可 以 直接 使 用 enum 中 的 元 素 名 ) ,下面 的 示例 以 statie 的 方式 引入 Watercolors。 
这 个 示例 使 用 了 EnumSet， 这 是 Java SE5 中 的 新 工具 ， 用 来 从 enum 直 接 创建 Set。( 在 第 19 章 中 ， 
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我 们 会 详细 介绍 EnumSet。) 在 这 里 ， 我 们 向 static 方 法 EnumSetrangeO 传 人 某 个 范围 的 第 一 个 
元 素 与 最 后 一 个 元 素 ， 然 后 它 将 返回 一 个 Set， 其 中 包含 该 范围 内 的 所 有 元 素 : 


//: generics/WatercolorSets. java 

‘import generics.watercolors.*; 

import java.util.*; 

import static net.mindview.util.Print.*; 
import static net.mindview.util.Sets.*; 

import static generics.watercolors Watercolors. 





public class Watercolorsets { 
public static void main(String{] args) { 
Set<Watercolors> setl = 
EnumSet. range (BRILLIANT_RED, VIRIDIAN_HUE); 
Set<Watercolors> set2 = 

















[642 EnumSet . range (CERULEAN_BLUE_HUE, BURNT_UMBER) ; 
print("set1: ”+ set1); 
print("set2: " + set2); 


print("union(setl, set2): " + union(setl, set2)); 
Set<Watercolors> subset = intersection(setl. set2); 
print("intersection(set1, set2): ”+ subset); 
Print("difference(setl, subset): " + 
difference(setl, subset)); 
print("difference(set2, subset): ”+ 
difference(set2, subset)): 
Print("complement(setl, set2): ”+ 
complement (set, set2)); 
} 
} /* Output: (Sample) 
setl: [BRILLIANT_RED, CRIMSON, MAGENTA, ROSE_MADDER, 
VIOLET, CERULEAN_BLUE_HUE, PHTHALO_BLUE, ULTRAMARINE, 
COBALT_BLUE_HUE, PERMANENT_GREEN, VIRIDIAN_HUE] 
set2: TCERULEAN BLUE_HUE, PHTHALO_BLUE, ULTRAMARINE, 
COBALT_BLUE_HUE, PERMANENT_GREEN, VIRIDIAN HUE, SAP GREEN, 
YELLOW_OCHRE, BURNT_SIENNA, RAW_UMBER, BURNT_UMBER] 
union(set1, set2): [SAP_GREEN, ROSE_MADDER, YELLOW_OCHRE, 
PERMANENT GREEN, BURNT_UMBER, COBALT_BLUE_HUE, VIOLET, 
BRILLIANT_RED, RAW_UMBER, ULTRAMARINE, BURNT_SIENNA, 
CRIMSON, CERULEAN BLUE HUE, PHTHALO_BLUE, MAGENTA, 
VIRIDIAN_HUE] 
intersection(set1, set2): [ULTRAMARINE, PERMANENT_GREEN, 
COBALT_BLUE_HUE, PHTHALO_ BLUE, CERULEAN_BLUE_HUE, 
VIRIDIAN_HUE] 
difference(seti, subset): [ROSE_MADDER, CRIMSON, VIOLET, 
MAGENTA, BRILLIANT_RED) 
difference(set2, subset): [RAW_UMBER, SAP_GREEN, 
YELLOW_OCHRE, BURNT_STENNA, BURNT_UMBER} 
complement (set1, set2): [SAP_GREEN, ROSE_MADDER, 
YELLOW_OCHRE, BURNT_UMBER, VIOLET, BRILLIANT_RED, 
RAW_UMBER, BURNT_SIENNA, CRIMSON, MAGENTA) 
#7:~ 


我 们 可 以 从 输出 中 看 到 各 种 关系 运算 的 结果 。 
下 面 的 示例 使 用 Sets.difference0 打 印 出 java.util 包 中 各 种 Collection 类 与 Map 类 之 间 的 方法 
差异 : 


//: net/mindview/util/ContainerMethodDi fferences. java 
package net.mindview.util; 
import java.lang.reflect.*; 
import java.util.*; 
Public class ContainerMethodDifferences { 
static Set<String> methodSet(Class<?> type) { 
Set<String> result = new TreeSet<String>(); 
for(Method m : type.getMethods()) 
result. add(m.getName()); 








| 
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return result; 
} 
static void interfaces(Class<?> type) { 
System.out.print("Interfaces in ”+ 
type.getSimpleName() + ": "); 
List<String> result = new ArrayList<String>(); 
for(Class<?> c : type.getInterfaces()) 
result.add(c.getSimpleName()); 
System.out.printin(result) : 
} 
static Set<String> object = methodSet (Object.class); 
static { object.add("clone"); } 
static void 
difference(Class<?> superset, Class<?> subset) { 
System.out.print (superset. getSimpleName() + 
“ extends " + subset.getSimpleName() + ", adds: "); 
Set<String> comp = Sets.difference( 
methodSet (superset), methodSet (subset)); 
comp. removeAll (object): // Don't show ‘Object’ methods 
System.out.printin(comp) ; 
interfaces (superset) ; 
} 
public static void main(String[] args) { 
System.out.printin("Collection: ”+ 
methodSet (Collection.class)); 
interfaces (Collection.class); 
difference(Set.class, Collection.class) ; 
difference(HashSet.class, Set.class); 
difference(LinkedHashSet.class, HashSet.class): 
difference(TreeSet.class, Set.class); 
difference(List.class, Collection.class) 
difference(ArrayList.class, List.class: 
difference(LinkedList.class, List.class); 
difference(Queve.class, Collection.class): 
difference (PriorityQueue.class, Queue.class); 
System.out.printin("Map: ”+ methodSet(Map.class)); 
difference(HashMap.class, Hap.class) ; 
difference(LinkedHashMap.class, HashMap.class); [644] 
difference(SortedMap.class, Map.class); 
difference(TreeMap.class, Map.class) ; 
} 
} Mi~ 


在 第 11 章 的 “总 结 ”中 ， 我 们 使 用 了 这 个 程序 的 输出 结果 。 

练习 17，(4) 研究 IDK 文 档 中 有 关 EnumSet 的 部 分 ， 你 会 看 到 它 定义 了 clone() 方 法 。 然 而 ， 
在 Setsjava 中 ， 你 却 不 能 复制 Set 接 口中 的 引用 。 请 试 着 修改 Setsjava， 使 其 不 但 能 接受 一 般 的 
Set 接 口 ， 而 且 能 直接 接受 EnumSet， 并 使 用 clone0 而 不 是 创建 新 的 HashSet 对 象 。 


15.5 匿名 内 部 类 


泛 型 还 可 以 应 用 于 内 部 类 以 及 匿名 内 部 类 。 下 面 的 示例 使 用 匿名 内 部 类 实现 了 Generator 
接口 : 


//: generics/BankTeller. java 
// A very simple bank teller simulation. 
import java.util.*; 

import net.mindview.util.*; 














class Customer { 
private static long counter 
private final long id = counter++; 
private Customer() {} 
public String toString() { return “Customer "+ id: } 
// A method to produce Generator objects: 
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public static Generator<Customer> generator() { 
return new Generator<Customer>() { 
public Customer next() { return new Customer(): } 





} 


class Teller { 

Private static long counter = 1; 

private final long id = counter++; 

private Teller() {} 

public String toString() { return "Teller "+ id; } 

// A single Generator object: 

public static Generator<Teller> generator = 

new Generator<Teller>() { 

public Teller next() { return new Teller(); } 
X: 
} 


public class BankTeller { 
public static void serve(Teller t, Customer c) { 
System.out.printin(t +" serves " + c); 


} 

Public static void main(Strfng[] args) { 
Random rand = new Random(47); 
Queue<Customer> Line = new LinkedList<Customer>(); 
Generators. fill(line, Customer.generator(), 15); 
ListcTeller> tellers = new ArrayList<Teller>(); 
Generators. fill(tellers, Teller.generator, 4); 
for(Customer c : line) 

serve(tellers.get(rand.nextInt(tellers.size())), c); 


} 
} /* Output: 
Teller 3 serves Customer 


1 
Teller 2 serves Customer 2 
Teller 3 serves Customer 3 
Teller 1 serves Customer 4 
Teller 1 serves Customer 5 
Teller 3 serves Customer 6 
Teller 1 serves Customer 7 
Teller 2 serves Customer 8 
Teller 3 serves Customer 9 
Teller 3 serves Customer 10 
Teller 2 serves Customer 11 
Teller 4 serves Customer 12 
Teller 2 serves Customer 13 
Teller 1 serves Customer 14 
Teller 1 serves Customer 15 


Whim 


Customer 和 Teller 类 都 只 有 private 的 构造 器 ， 这 可 以 强制 你 必须 使 用 Generator 对 象 。 
Customer 有 一 个 generator( 方 法 ， 每 次 执行 它 都 会 生成 一 个 新 的 Generator<Customer> 对 象 。 
我 们 其 实 不 需要 多 个 Generator 对 象 ，Teller 就 只 创建 了 一 个 public 的 generator 对 象 。 在 main0 方 
法 中 可 以 看 到 ， 这 两 种 创建 Generator 的 方式 都 在 人 IO 中 用 到 了 。 

由 于 Customer 中 的 generator( 方 法 ， 以 及 Teller 中 的 Generator 对 象 都 声明 成 了 static 的 ， 所 
以 它们 无 法 作为 接口 的 一 部 分 ， 因 此 无 法 用 接口 这 种 特定 的 惯用 法 来 泛 化 这 二 者 。 尽 管 如 此 ， 
它们 在 fi10 方 法 中 都 工作 得 很 好 。 

在 第 21 章 中 ， 我 们 还 会 看 到 关于 这 个 排队 问题 的 另 一 个 版 本 。 

练习 18: (3) 遵循 BackTellerjava 的 形式 ， 创 建 一 个 Ocean 中 BigFish 吃 LittieFish 的 例子 。 
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15.6 构建 复杂 模型 


证 型 的 一 个 重要 好 处 是 能 够 简单 而 安全 地 创建 复杂 的 模型 。 例 如 ， 我 们 可 以 很 容易 地 创建 
List 元 组 : 

//: generics/TupteList.java 

// Combining generic types to make complex generic types. 


import java.util.*; 
import net.mindview.util.*; 


public class TupteList<A,B,C,D> 
extends ArrayList<FourTuple<A,B,C,D>> { 
public static void main(String[] args) { 
TupleList<Vehicle, Amphibian, String, Integer> tl = 
new TupleList<Vehicle, Amphibian, String, Integer>(); 
tl.add(TupleTest.n()); 
tl.add(TupleTest.n()); 
for (FourTuple<Vehicle, Amphibian. String, Integer> i: tl) 
System.out.printin(i): 
} 
) /* Output: (75% match) 
(Vehicle@11b86e7, Amphibian@3sce36, hi, 47) 
(Vehicte@757aef, Amphibianed9f9c3, hi, 47) 
Whe 


尽管 这 看 上 去 有 些 宛 长 〈 特 别 是 迭代 器 的 创建 )， 但 最 终 还 是 没有 用 过 多 的 代码 就 得 到 了 一 
个 相当 强大 的 数据 结构 。 

下 面 是 另 一 个 示例 ， 它 展示 了 使 用 泛 型 类 型 来 构建 复杂 模型 是 多 么 的 简单 。 即 使 每 个 类 都 
是 作为 一 个 构建 块 创建 的 ， 但 是 其 整个 还 是 包含 许多 部 分 。 在 本 例 中 ， 构 建 的 模型 是 一 个 零售 
店 ， 它 包含 走廊 、 货 架 和 商品 : 


/1: generics/Store.java 

// Building up a complex model using generic containers. 
import java.util.*; 

import net.mindview.util.*; 





class Product { 
private final int id: 
private String description; 
private double price; 
public Product(int IDnumber, String descr, double price){ 
id = IDnumber; 
description = descr; 
this.price = price; 
System.out.printin(toString()); 
} 
public String toString() { 
return id + ": * + description + *, price: $" + price; 
} 
public void priceChange(double change) { 
price += change; 
} 
public static Generator<Product> generator = 
new Generator<Product>() { 
private Random rand = new Random(47) ; 
public Product next() { 
return new Product (rand.nextInt(109@), "Test", 
Math.round(rand.nextDouble() * 1860.8) + 6.99); 


2 


class Shelf extends ArrayList<Product> { 
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public Shelf (int nProducts) { 
Generators. fill(this, Product.generator, nProducts) ; 
$ 
} 


class Aisle extends ArrayList<shelf> { 

public Aisle(int nShelves, int nProducts) { 
for(int i = 0; i < nShelves; i++) 
add(new Shelf (nProducts)): 

} 

} 

class CheckoutStand {} 

class Office {} 





public class Store extends ArrayList<Aisle> { 
private ArrayList<CheckoutStand> checkouts = 
new ArrayList<CheckoutStand>() ; 
private Office office = new Office(); 
Public Store(int nAisles, int nShelve: 
for(int 1 = 6; i < nAisles; i++) 
add(new Aiste(nShelves, nProducts)); 





int mProducts) { 


} 
public String toString() { 
StringBuilder result = new StringBuilder(); 
for(Aisle a : this) 
for(Shelf s : a) 
for(Product p : s) { 
result. append(p); 
result. append("\n"); 





return result.toString(): 


} 
public static void main(String] args) { 
System.out.printin(new Store(14, 5, 10)); 


4 
} /* Output: 


258: Test, price: $400.99 
861: Test. price: $160.99 
868: Test, price: $417.99 
207: Test, price: $268.99 
551: Test, price: $114.99 


Test, price: $804.99 
Test, price: $554.99 
140: Test, price: $530.99 





Wh 
正如 我 们 在 Store.toString0 中 看 到 的 ， 其 结果 是 许多 层 容器 ， 但 是 它们 是 类 型 安全 且 可 管理 


的 。 令 人 印象 深刻 之 处 是 组 装 这 个 的 模型 十 分 容易 ， 并 不 会 成 为 智力 挑战 。 
(649) 练习 19: (2) 遵循 Storejava 的 形式 ， 构 建 一 个 容器 化 的 货船 模型 。 


15.7 擦 除 的 神秘 之 处 


当 你 开始 更 深入 地 钻研 泛 型 时 ， 会 发 现 有 大 量 的 东西 初 看 起 来 是 没有 意义 的 。 例 如 ， 尽 管 
可 以 声明 ArrayList.class， 但 是 不 能 声明 ArrayList<Integer>.class。 请 考虑 下 面 的 情况 : 


//: generics/ErasedTypeEquivalence. java 
import java.util.*; 














Public class ErasedTypeEquivalence { 
public static void main(String[] args) { 
Class cl = new ArrayList<String>().getClass(); 
Class c2 = new ArrayList<Integer>().getClass(); 
System.out.printin(cl == c2); 
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} /* Output: 
true 
Whi 


ArrayList<String> 和 ArrayList<Integer> 很 容易 被 认为 是 不 同 的 类 型 。 不 同 的 类 型 在 行为 方 
面 肯定 不 同 ， 例 如 ， 如 果 举 试 着 将 一 个 Integer 放 入 ArrayList<String>， 所 得 到 的 行为 〈 将 失败 ) 
与 把 一 个 Integer 放 人 ArrayList<Integer> (将 成 功 ) 所 得 到 的 行为 完全 不 同 。 但 是 上 面 的 程序 会 
认为 它们 是 相同 的 类 型 。 

下 面 是 的 示例 是 对 这 个 谜 题 的 一 个 补充 : 

//: generics/LostInformation.java 

import java.util.*; 


class Frob {} 

class Fnorkle {} 

class Quark<Q> {} 

class Particle<POSITION,MOMENTUM> {} 


public class LostInformation { 
public static void main(String(] args) { 
List<Frob> list = new ArrayList<Frob>(); 
Map<Frob,Fnorkle> map = new HashMap<Frob,Fnorkle>(); 
Quark<Fnorkle> quark = new Quark<Fnorkle>(); 
Particle<Long,Double> p = new Particle<Long. Double>(); 
System.out.printin (Arrays. toString( 
list.getClass().getTypeParameters())); 
System. out.printin (Arrays. toString( 
map.getClass() .getTypeParameters())); 
System. out .printin(Arrays. toString( 
quark. getClass().getTypeParameters())): 
System. out.printin(Arrays.toString( 
p.getClass().getTypeParameters())); 


} 
} /* Output: 
[E] 
{K, VI 


[Q] 
[POSITION, MOMENTUM] 
Whim 


根据 JDK 文 档 的 描述 ，Class.getTypeParameters0 将 “返回 一 个 TypeVariable 对 象 数组 ， 表 
示 有 泛 型 声明 所 声明 的 类 型 参数 ……” 这 好 像 是 在 暗示 你 可 能 发 现 参数 类 型 的 信息 ， 但 是 ， 正 
如 你 从 输出 中 所 看 到 的 ， 你 能 够 发 现 的 只 是 用 作 参 数 占 位 符 的 标识 符 ， 这 并 非 有 用 的 信息 。 

因此 ， 残 酷 的 现实 是 : 

在 泛 型 代码 内 部 ， 无 法 获得 任何 有 关 泛 型 参数 类 型 的 信息 。 

因此 ， 你 可 以 知道 诸如 类 型 参数 标识 符 和 泛 型 类 型 边界 这 类 的 信息 一 一 你 却 无 法 知道 用 来 
创建 某 个 特定 实例 的 实际 的 类 型 参数 。 如 果 你 曾经 是 C++ 程序 员 ， 那 么 这 个 事实 肯定 让 你 觉得 
很 诅 形 ， 在 使 用 Java 泛 型 工作 时 它 是 必须 处 理 的 最 基本 的 问题 。 

Java 泛 型 是 使 用 擦 除 来 实现 的 ， 这 意味 着 当 你 在 使 用 泛 型 时 ， 任 何 具体 的 类 型 信息 都 被 控 
除了 ， 你 唯一 知道 的 就 是 你 在 使 用 一 个 对 象 。 因 此 List<String> 和 List<Integer> 在 运行 时 事实 上 
是 相同 的 类 型 。 这 两 种 形式 都 被 擦 除 成 它们 和 的“ 原生” 类型， 即 List。 理 解 擦 除 以 及 应 该 如 何 处 
理 它 ， 是 你 在 学 习 Java 泛 型 时 面临 的 最 大 障碍 ， 这 也 是 我 们 在 本 节 将 要 探讨 的 内 容 。 

15.7.1 C++ 的 方式 
| 下 面 是 使 用 模版 的 C++ 示例 ， 你 将 注意 到 用 于 参数 化 类 型 的 语法 十 分 相似 ， 因 为 Java 是 受 
C++ 的 启发 ， 
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//: generics/Templates.cpp 
#include <iostream> 
using namespace std; 


template<class T> class Manipulator { 
T obj; 

public: 
Manipulator(T x) { obj = x: } 
void manipulate() { obj.f(); } 

oH 


class HasF { 
public: 

void f() { cout << "HasF::f()" << endl; } 
k 


int main() { 
HasF hf; 
Manipulator<HasF> manipulator (hf) ; 
manipulator .manipulate() ; 

} /* Output: 

HasF::f( 

Whim 


Manipulator 类 存储 了 一 个 类 型 T 的 对 象 ， 有 意思 的 地 方 是 manipulate0 方 法 ， 它 在 obj 上 调 
用 方法 f0。 它 怎么 能 知道 f0 方 法 是 为 类 型 参数 T 而 存在 的 呢 ? 当 你 实例 化 这 个 模版 时 ，C++ 编 译 
器 将 进行 检查 ， 因 此 在 Manipulator<HasF> 被 实例 化 的 这 一 刻 ， 它 看 到 HasF 拥 有 一 个 方法 f0。 
如 果 情 况 并 非 如 此 ， 就 会 得 到 一 个 编译 期 错误 ， 这 样 类 型 安全 就 得 到 了 保障 。 

用 C++ 编写 这 种 代码 很 简单 ， 因 为 当 模版 被 实例 化 时 ， 模 版 代码 知道 其 模版 参数 的 类 型 。 
Java 泛 型 就 不 同 了 。 下 面 是 HasF 的 Java 版 本 : 


//: Benerics/HasF.java 


public class HasF { 
public void f() { System.out.printin("HasF.f()"); } 


} Mi~ 
如 果 我 们 将 这 个 示例 的 其 余 代码 都 翻译 成 Java， 那 么 这 些 代码 将 不 能 编译 : 


//: generics/Manipulation.java 
// {CompileTimeError} (Won't compile) 


class Manipulator<T> { 
private T obj; 
public Manipulator(T x) { obj = x; } 
7/ Error: cannot find symbol: method fO): 
public void manipulate() { obj.f(); } 

} 


public class Manipulation { 
public static void main(String{] args) { 
HasF hf = new HasF(); 
Manipulator<HasF> manipulator = 
new Manipulator<HasF> (hf) : 
manipulator .manipulate(); 





} 
} Mi~ 


由 于 有 了 擦 除 ，Java 编 译 器 无 法 将 manipulate0) 必 须 能 够 在 obj 上 调用 f0 这 一 需求 映射 到 
HasF 拥 有 f0 这 一 事实 上 。 为 了 调用 f0， 我 们 必须 协助 泛 型 类 ， 给 定 泛 型 类 的 边界 ， 以 此 告知 纺 
译 器 只 能 接受 遵循 这 个 边界 的 类 型 。 这 里 重用 了 extends 关 键 字 。 由 于 有 了 边界 ， 下 面 的 代码 就 
可 以 编译 了 : 





Žž g 375 





//: generics/Manipulator2. java 


class Manipulator2<T extends HasF> { 
private T obj; 
public Manipulator2(T x) { obj = x; } 
public void manipulate() { obj.f(); } 
} Mi~ 


边界 <T extends HasF> 声 明 T 必 须 具 有 类 型 HasF 或 者 从 HasF 导 出 的 类 型 。 如 果 情 况 确实 如 
此 ， 那 么 就 可 以 安全 地 在 obj 上 调用 f0 了 。 

我 们 说 泛 型 类 型 参数 将 擦 除 到 它 的 第 一 个 边界 〈 它 可 能 会 有 多 个 边界 ， 稍 候 你 就 会 看 到 ) ， 
我 们 还 提 到 了 类 型 参数 的 擦 除 。 编 译 器 实际 上 会 把 类 型 参数 替换 为 它 的 擦 除 ， 就 像 上 面 的 示例 
一 样 。T 擦 除 到 了 HasF， 就 好 像 在 类 的 声明 中 用 HasF 替 换 了 T 一 样 。 

你 可 能 已 经 正确 地 观察 到 ， 在 Manipulation2.java 中 ， 泛 型 没有 贡献 任何 好 处 。 只 需 很 容易 
地 自己 去 执行 擦 除 ， 就 可 以 创建 出 没有 泛 型 的 类 : 

//: generics/Manipulator3. java 

class Manipulator3 { 

private HasF obj; . 
public Manipulator3(HasF x) { obj = x; } 


public void maniputate() { obj.f(); } 
} Mi~ 


这 提出 了 很 重要 的 一 点 : 只 有 当 你 希望 使 用 的 类 型 参数 比 某 个 具体 类 型 (以 及 它 的 所 有 子 
类 型 ) 更 加 “ 泛 化 ”时 一 一 也 就 是 说 ， 当 你 希望 代码 能 够 跨 多 个 类 工作 时 ， 使 用 泛 型 才 有 所 帮 
助 。 因 此 ， 类 型 参数 和 它们 在 有 用 的 泛 型 代码 中 的 应 用 ， 通 常 比 简单 的 类 替换 要 更 复杂 。 但 是 ， 
不 能 因此 而 认为 <T extends HasF> 形 式 的 任何 东西 而 都 是 有 缺陷 的 。 例 如 ， 如 果 某 个 类 有 一 个 
返回 T 的 方法 ， 那 么 泛 型 就 有 所 帮助 ， 因 为 它们 之 后 将 返回 确切 的 类 型 : 


//: generics/ReturnGenericType.java 


class ReturnGenericType<T extends HasF> { 
private T obj; 
Public ReturnGenericType(T x) { obj = x; } 
public T get() { return obj; } 

} Mi~ 


必须 查看 所 有 的 代码 ， 并 确定 它 是 否 “ 足 够 复杂 ”到 必须 使 用 泛 型 的 程度 。 

我 们 将 在 本 章 稍 后 介绍 有 关 边 界 的 更 多 细节 。 

练习 20:(1) 创建 一 个 具有 两 个 方法 的 接口 ， 以 及 一 个 实现 了 这 个 接口 并 添加 了 另 一 个 方法 的 
类 。 在 另 一 个 类 中 ， 创 建 一 个 泛 型 方法 ， 它 的 参数 类 型 由 这 个 接口 定义 了 边界 ， 并 展示 该 接口 中 
的 方法 在 这 个 泛 型 方法 中 都 是 可 调用 的 。 在 main0 方 法 中 传递 一 个 实现 类 的 实例 给 这 个 泛 型 方法 。 
15.7.2 迁移 兼容 性 

为 了 减少 潜在 的 关于 擦 除 的 混淆 ， 你 必须 清楚 地 认识 到 这 不 是 一 个 语言 特性 。 它 是 Java 的 
泛 型 实现 中 的 一 种 折 中 ， 因 为 泛 型 不 是 Java 语 言 出 现时 就 有 的 组 成 部 分 ， 所 以 这 种 折 中 是 必需 
的 。 这 种 折 中 会 使 你 痛苦 ， 因 此 你 需要 习惯 它 并 了 解 为 什么 它 会 是 这 样 。 

如 果 泛 型 在 Java 1.0 中 就 已 经 是 其 一 部 分 了 ， 那 么 这 个 特性 将 不 会 使 用 控 除 来 实现 一 一 它 将 
使 用 具体 化 ， 使 类 型 参数 保持 为 第 一 类 实体 ， 因 此 你 就 能 够 在 类 型 参数 上 执行 基于 类 型 的 语言 
操作 和 反射 操作 。 你 将 在 本 章 稍 后 看 到 ， 按 除 减 少 了 泛 型 的 泛 化 性 。 泛 型 在 Java 中 仍旧 是 有 用 
的 ， 只 是 不 如 它们 本 来 设想 的 那么 有 用 ， 而 原因 就 是 控 除 。 

在 基于 擦 除 的 实现 中 ， 泛 型 类 型 被 当 作 第 二 类 类 型 处 理 ， 即 不 能 在 某 些 重要 的 上 下 文 环境 
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中 使 用 的 类 型 。 泛 型 类 型 只 有 在 静态 类 型 检查 期 间 才 出 现 ， 在 此 之 后 ,程序 中 的 所 有 泛 型 类 型 
都 将 被 擦 除 ， 蔡 换 为 它们 的 非 泛 型 上 界 。 例 如 ， 诸 如 List<T> 这 样 的 类 型 注解 将 被 擦 除 为 List， 
而 普通 的 类 型 变量 在 未 指定 边界 的 情况 下 将 被 擦 除 为 Object。 

擦 除 的 核心 动机 是 它 使 得 泛 化 的 客户 端 可 以 用 非 泛 化 的 类 库 来 使 用 ， 反 之 亦 然 ， 这 经 常 被 
称 为 “迁移 兼容 性 "。 在 理想 情况 下 ， 当 所 有 事物 都 可 以 同时 被 泛 化 时 ， 我 们 就 可 以 专注 于 此 。 
在 现实 中 ， 即 使 程序 员 只 编写 泛 型 代码 ， 他 们 也 必须 处 理 在 Java SE5 之 前 编写 的 非 泛 型 类 库 。 那 
些 类 库 的 作者 可 能 从 没有 想 过 要 泛 化 它们 的 代码 ,或 者 可 能 刚刚 开始 接触 泛 型 。 

因此 Java 泛 型 不 仅 必须 支持 向 后 兼容 性 ， 即 现 有 的 代码 和 类 文件 仍旧 合法 ， 并 且 继 续 保持 
其 之 前 的 含义 ， 而且 还 要 支持 迁移 兼容 性 ， 使 得 类 库 按照 它们 自己 的 步调 变 为 泛 型 的 ， 并 且 当 
某 个 类 库 变 为 泛 型 时 ， 不 会 破坏 依赖 于 它 的 代码 和 应 用 程序 。 在 决定 这 就 是 目标 之 后 ，Java 设 
计 者 们 和 从 事 此 问题 相关 工作 的 各 个 团队 决策 认为 擦 除 是 唯一 可 行 的 解决 方案 。 通 过 允许 非 泛 
型 代码 与 泛 型 代码 共存 ， 擦 除 使 得 这 种 向 着 泛 型 的 迁移 成 为 可 能 。 

例如 ， 假 设 某 个 应 用 程序 具有 两 个 类 库 X 和 Y， 并 且 Y 还 要 使 用 类 库 Z。 随 着 Java SE5 的 出 现 ， 
这 个 应 用 程序 和 这 些 类 库 的 创建 者 最 终 可 能 希望 迁移 到 泛 型 上 。 但 是 ， 当 进行 这 种 迁移 时 ,他 
们 有 着 不 同 动机 和 限制 。 为 了 实现 迁移 兼容 性 ， 每 个 类 库 和 应 用 程序 都 必须 与 其 他 所 有 的 部 分 
是 否 使 用 了 泛 型 无 关 。 这 样 ， 它 们 必须 不 具备 探测 其 他 类 库 是否 使 用 了 泛 型 的 能 力 。 因 此 ， 某 
个 特定 的 类 库 使 用 了 泛 型 这 样 的 证 据 必 须 被 “ 擦 除 ”。 

如 果 没有 某 种 类 型 的 迁移 途径 ， 所 有 已 经 构建 了 很 长 时 间 的 类 库 就 需要 与 希望 迁移 到 Java 
泛 型 上 的 开发 者 们 说 再 见 了 。 但 是 ， 类 库 是 编程 语言 无 可 争议 的 一 部 分 ， 它 们 对 生产 效率 会 产 
生 最 重要 的 影响 ， 因 此 这 不 是 一 种 可 以 接受 的 代价 。 擦 除 是 否 是 最 佳 的 或 者 唯一 的 迁移 途径 ， 
还 需要 时 间 来 证 明 。 

15.7.3 擦 除 的 问题 

因此 ， 擦 除 主要 的 正当 理由 是 从 非 泛 化 代码 到 泛 化 代码 的 转变 过 程 ， 以 及 在 不 破坏 现 有 类 
库 的 情况 下 ， 将 泛 型 融入 Java 语 言 。 擦 除 使 得 现 有 的 非 泛 型 客户 端 代码 能 够 在 不 改变 的 情况 下 
继续 使 用 ， 直 至 客户 端 准备 好 用 泛 型 重 写 这 些 代 码 。 这 是 一 个 崇高 的 动机 ， 因 为 它 不 会 突然 间 
破坏 所 有 现 有 的 代码 。 

控 除 的 代价 是 显著 的 。 泛 型 不 能 用 于 显 式 地 引用 运行 时 类 型 的 操作 之 中 ， 例 如 转型 
instanceof 操 作 和 new 表 达 式 。 因 为 所 有 关于 参数 的 类 型 信息 都 丢失 了 ， 无 论 何 时 ， 当 你 在 编写 
泛 型 代码 时 ， 必 须 时 刻 提醒 自己 ， 你 只 是 看 起 来 好 像 拥有 有 关 参 数 的 类 型 信息 而 已 。 因此， 如 
果 你 编写 了 下 面 这 样 的 代码 段 


class FoosT> { 
T var; 


} 

那么 ， 看 起 来 当 你 在 创建 Foo 的 实例 时 : 

FooxCat> f = new Foo<Cat>(); 
class Foo 中 的 代码 应 该 知道 现在 工作 于 Cat 之 上 ， 而 泛 型 语法 也 在 强烈 暗示 在 整个 类 中 的 各 个 
地 方 ， 类 型 都 在 被 替换 。 但 是 事实 并 非 如 此 ， 无 论 何 时 ， 当 你 在 编写 这 个 类 的 代码 时 ， 必 须 提 
醒 自己 :“ 不 ， 它 只 是 一 个 Object。” 

另外 ， 擦 除 和 迁移 兼容 性 意味 着 ,使 用 泛 型 并 不 是 强制 的 ， 尽 管 你 可 能 希望 这 样 : 

/1: generics/ErasureAndInheri tance. java 


class GenericBase<T> { 
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private T element; 
public void set(T arg) { arg = element: } 
public T get() { return element; } 

} 


class Derivedi<T> extends GenericBase<T> {} 
class Derived2 extends GenericBase {} // No warning 


// class Derived3 extends GenericBase<?> {} 

// Strange error: 

// unexpected type found : ? 

// required: class or interface without bounds 


public class ErasureAndInheritance { 
@SuppressWarnings ("unchecked") 
public static void main(String[] args) { 
Derived? d2 = new Derived2(); 
Object obj = d2.get(); 
d2.set(obj); // Warning here! 
} Mine 
Derived2 继 承 自 GenericBase， 但 是 没有 任何 泛 型 参数 ， 而 编译 器 不 会 发 出 任何 警告 。 警 告 
在 set0 被 调用 时 才 会 出 现 。 
为 了 关闭 警告 ，Java 提 供 了 一 个 注解 ， 我 们 可 以 在 列表 中 看 到 它 〔 这 个 注解 在 Java SE5 之 前 
的 版 本 中 不 支持 ): 


@SuppressWarnings ("unchecked") 

注意 ， 这 个 注解 被 放置 在 可 以 产生 这 类 警告 的 方法 之 上 ， 而 不 是 整个 类 上 。 当 你 要 关闭 警 
告 时 ， 最 好 是 尽量 地 “ 聚 售 "， 这 样 就 不 会 因为 过 于 宽泛 地 关闭 警告 ， 而 导致 意外 地 遮蔽 掉 真 正 
的 问题 。 

可 以 推断 ，Derived3 产 生 的 错误 意味 着 编译 器 期 望 得 到 一 个 原生 基 类 。 

当 你 希望 将 类 型 参数 不 要 仅仅 当 作 Objeet 处 理 时 ， 就 需要 付出 额外 努力 来 管理 边界 ， 并 且 
与 在 Ct++、Ada 和 Eiffel 这 样 的 语言 中 获得 参数 化 类 型 相 比 ， 你 需要 付出 多 得 多 的 努力 来 获得 少 得 
多 的 回报 。 这 并 不 是 说 ， 对 于 大 多 数 的 编程 问题 而 言 ， 这 些 语言 通常 都 会 比 Java 更 得 心 应 手 ， 
这 只 是 说 ， 它 们 的 参数 化 类 型 机 制 比 Java 的 更 灵活 、 更 强大 。 

15.7.4 ”边界 处 的 动作 

正 是 因为 有 了 擦 除 ， 我 发 现 泛 型 最 令 人 困惑 的 方面 源 自 这 样 一 个 事实 ， 即 可 以 表示 没有 任 

何 意义 的 事物 。 例 如 : 


//: generics/ArrayMaker .java 
import java.lang.reflect.*; 
import java.util.*; 


public class ArrayMaker<T> { 
private Class<T> kind; 
public ArrayMaker(Class<T> kind) { this.kind = kind: } 
@SuppressWarnings ("unchecked") 
TU create(int size) { 
return (T[])Array.newInstance(kind, size); 


} 
Public static void main(String[] args) { 
ArrayMaker<String> stringMaker = 
new ArrayMaker<String>(String.class) ; 
String(] stringArray = stringMaker.create(9) ; 
System.out.printIn (Arrays. toString(stringArray)): 
} 











658 








378 KISH 





} /* Output: 
(null, null, null, null, null, null, null, null, null) 
"Whim 


即使 kind 被 存储 为 Class<T>， 擦 除 也 意味 着 它 实 际 将 被 存储 为 Class， 没 有 任何 参数 。 因 此 ， 
当 你 在 使 用 它 时 ， 例 如 在 创建 数组 时 ，ArraynewInstance0 实 际 上 并 未 拥有 kind 所 蕴含 的 类 型 信 
息 ， 因 此 这 不 会 产生 具体 的 结果 ， 所 以 必须 转型 ， 这 将 产生 一 条 令 你 无 法 满意 的 警告 。 

注意 ， 对 于 在 泛 型 中 创建 数组 ， 使 用 Array.newlInstance0 是 推荐 的 方式 。 

如 果 我 们 要 创建 一 个 容器 而 不 是 数组 ， 情 况 就 有 些 不 同 了 : 

/1/: generics/ListMaker. java 

import java.util.*; 


public class ListMaker<T> { 
List<T> create() { return new ArrayList<T>(); } 
public static void main(String[] args) { 
ListMaker<String> stringMaker= new ListMaker<String>(); 
List<String> stringList = stringMaker.create(); 


} 

dh 

编译 器 不 会 给 出 任何 警告 ， 尽 管 我 们 (从 擦 除 中 ) 知道 在 create0 内 部 的 new ArrayList<T> 
中 的 <T> 被 移 除 了 一 一 在 运行 时 ， 这 个 类 的 内 部 没有 任何 <T>， 因 此 这 看 起 来 毫 无 意义 。 但 是 如 
果 你 遵从 这 种 思路 ， 并 将 这 个 表达 式 改 为 new ArrayList0， 编 译 器 就 会 给 出 警告 。 

在 本 例 中 ， 这 是 否 真 的 毫 无 意义 呢 ? 如 果 返 回 list 之 前 ， 将 某 些 对 象 放 入 其 中 ， 就 像 下 面 这 
样 ， 情 况 又 会 如 何 呢 ? 

//: generics/FilledListMaker. java 

‘import java.util.*; 


public class FilledListMaker<T> { 
List<T> create(T t, int n) { 
List<T> result = new ArrayList<T>(); 
for(int i = 6; i < n; i++) 
result.add(t); 
return result; 


public static void main(String[] args) { 
FilledListMaker<String> stringMaker = 
new FilledListtaker<String>(); 
List<String> list = stringMaker.create("Hello”, 4); 
System. out.printin(list); 


} I Output: 

[Hello, Hello, Hello, Hello} 

Whim 

即使 编译 器 无 法 知道 有 关 create0 中 的 T 的 任何 信息 ， 但 是 它 仍旧 可 以 在 编译 期 确保 你 放置 
到 result 中 的 对 象 具有 T 类 型 ， 使 其 适合 ArrayList<T>。 因 此 ， 即 使 擦 除 在 方法 或 类 内 部 移 除了 
有 关 实际 类 型 的 信息 ， 编 译 器 仍旧 可 以 确保 在 方法 或 类 中 使 用 的 类 型 的 内 部 一 致 性 。 

因为 擦 除 在 方法 体 中 移 除 了 类 型 信息 ， 所 以 在 运行 时 的 问题 就 是 边界 ， 即 对 象 进入 和 离开 
方法 的 地 点 。 这 些 正 是 编译 器 在 编译 期 执行 类 型 检查 并 插入 转型 代码 的 地 点 。 请 考虑 下 面 的 非 
泛 型 示例 : 


//: generics/SimpleHolder. java 


public class SimpleHolder { 
private Object obj; 
public void set(Object obj) { this.obj = obj; } 
public Object get() { return obj; } 
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public static void main(String[{] args) { 
SimpleHolder holder = new SimpleHolder(); 
holder .set ("Item"); 
String s = (String)holder.get(); 
} 
i 


如 果 用 javap -c SimpleHolder 反 编译 这 个 类 ， 就 可 以 得 到 下 面 的 (经 过 编辑 的 ) AB: 


public void set(java.lang.Object); 








6: — aload_@ 
as aload_1 
2: putfield #2; //Field obj :Object; 
LE return 

public java.lang.Object get(); 
9: aload_@ 
1: getfield #2; //Field obj:0bject; 
4: areturn 


public static void main(java. tang.String[]): 

new #3; //class SimpleHolder 

dup 

invokespecial #4; //Method “<init>":(V 
astore_1 

aload_1 

lde #5; //String Item 

invokevirtual #6; //Method set: (Object: )V 
aload_1 

invokevirtual #7; //Method get: (Object; 
checkcast #8; //class java/lang/String 
21: astore_2 

22: return 


setO 和 get0 方 法 将 直接 存储 和 产生 值 ， 而 转型 是 在 调用 get0 的 时 侯 接受 检查 的 。 
现在 将 泛 型 合并 到 上 面 的 代码 中 : 


//: Benerics/GenericHolder.java 





Doom 





public class GenericHolder<T> { 
private T obj; 
public void set(T obj) { this.obj = obj; } 
public T get() { return obj; } 
public static void main(String[] args) { 
GenericHolder<String> holder = 
new GenericHolder<String>() ; 
holder .set ("Item"); 
String s = holder.get(); 


} 
} Mi~ 
从 get0 返 回 之 后 的 转型 消失 了 ， 但 是 我 们 还 知道 传递 给 set0 的 值 在 编译 期 会 接受 检查 。 下 
面 是 相关 的 字 节 码 ; 
public void set (java.lang.Object); 
8: aload_9 
1: aload_t 
2: putfield #2; //Field obj:Object; 
5: return 
public java.lang.Object get(); 
@: aload_6 
1: getfield #2; //Field obj:Object; 
4: areturn 


public static void main(java.lang.String{]); 
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new #3; //class GenericHolder 
dup 

invokespecial #4; //Method "<init>": OV 
astore_1 

aload 

tde #5; //String Item 

invokevirtual #6; //Method set: (Object; )V 
aload_1 

invokevirtual #7; //Method get: ()Object; 
checkcast #8; //class java/lang/String 
astore 2 

22: return 


所 产生 的 字 节 码 是 相同 的 。 对 进入 set0 的 类 型 进行 检查 是 不 需要 的 ， 因 为 这 将 由 编译 器 执 
行 。 而 对 从 get0 返 回 的 值 进行 转型 仍旧 是 需要 的 ， 但 这 与 你 自己 必须 执行 的 操作 是 一 样 的 一 此 
处 它 将 由 编译 器 自动 插入 ， 因 此 你 写 人 〈 和 读 取 ) 的 代码 的 噪声 将 更 小 。 

由 于 所 产生 的 get0 和 set0 的 字 节 码 相同 ， 所 以 在 泛 型 中 的 所 有 动作 都 发 生 在 边界 处 一 对 传 
递 进来 的 值 进行 额外 的 编译 期 检查 ， 并 插入 对 传递 出 去 的 值 的 转型 。 这 有 助 于 澄清 对 控 除 的 混 
消 ， 记 住 ,“ 边 界 就 是 发 生动 作 的 地 方 。 


15.8 擦 除 的 补偿 


正如 我 们 看 到 的 ， 擦 除 丢失 了 在 泛 型 代码 中 执行 某 些 操作 的 能 力 。 任 何在 运行 时 需要 知道 
确切 类 型 信息 的 操作 都 将 无 法 工作 : 


//: generics/Erased. java 
// {CompileTimeError} (Won't compile) 





public class Erased<T> { 
private final int SIZE = 100; 
public static void f(Object arg) { 


if(arg instanceof T) {} 11 Error 
T var = new TO); W Error 
TU array = new, T{SIZE]; 41 Error 


TU] array = (T)new Object (SIZE]; // Unchecked warning 


cal ) hie 
偶尔 可 以 绕 过 这 些 问 题 来 编程 ， 但 是 有 时 必须 通过 引入 类 型 标签 来 对 氛 除 进行 补偿 。 这 意 
味 着 你 需要 显 式 地 传递 你 的 类 型 的 Class 对 象 ， 以 便 你 可 以 在 类 型 表达 式 中 使 用 它 。 
例如 ， 在 前 面 示例 中 对 使 用 instanceof 的 尝试 最 终 失败 了 ， 因 为 其 类 型 信息 已 经 被 擦 除了 。 
如 果 引 入 类 型 标签 ， 就 可 以 转 而 使 用 动态 的 isinstance0: 


//: generics/ClassTypeCapture. java 





class Building {} 
class House extends Building {} 


public class ClassTypeCapture<T> { 
Class<T> kind; 
public ClassTypeCapture(Class<T> kind) { 
this. kind = kind; 


} 
public boolean f(Object arg) { 
return kind. isInstance(arg); 


} 
public static void main(String{} args) { 
ClassTypeCapture<Building> ctt1 = 
new ClassTypeCapture<Bui lding> (Building.class); 
System.out.printin(ctt1.f(new Building())): 
System.out.printin(cttt. f(new House())): 
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ClassTypeCapture<House> ctt2 = 

new ClassTypeCapture<House> (House. class); 
System.out.printin(ctt2.f(new Building())); 
System.out.printin(ctt2.f(new House())): 


$ 
} /* Output: 
true 
true 
false 
true 
Wh 


编译 器 将 确保 类 型 标签 可 以 匹配 泛 型 参数 。 

练习 21: (4) 修改 ClassTypeCapture.java， 添 加 一 个 Map<String,Class<?>>， 一 个 addType 
(String typename,Class<?>kind) 方 法 和 一 个 createNew(String typename) 方 法 。createNew0 将 产 
生 一 个 与 其 参数 字符 串 相关 联 的 类 的 新 实例 ， 或 者 产生 一 条 错误 消息 。 
15.8.1 创建 类 型 实例 

在 Erasedjava 中 对 创建 一 个 new TO 的 尝试 将 无 法 实现 ， 部 分 原因 是 因为 擦 除 ， 而 另 一 部 分 
原因 是 因为 编译 器 不 能 验证 T 具 有 默认 《无 参 ) 构造 器 。 但 是 在 C++ 中 ， 这 种 操作 很 自然 、 很 直 
观 ， 并 且 很 安全 〈 它 是 在 编译 期 受到 检查 的 ) ; 


/11: generics/InstantiateGenericType.cpp 
// C++, not Java! 


template<class T> class Foo { 
T x; // Create a field of type T 
T* y; // Pointer to T 

public: 

// Initialize the pointer 

Foot) { y = new TO; } 

X: 


class Bar {}; 


int maino { 

Foo<Bar> fb; 

Foo<int> fi; // ... and it works with primitives 
y~ 


Java 中 的 解决 方案 是 传递 一 个 工厂 对 象 ， 并 使 用 它 来 创建 新 的 实例 。 最 便利 的 工厂 对 象 就 是 
Class 对 象 ， 因 此 如 果 使 用 类 型 标签 ， 那 么 你 就 可 以 使 用 newInstance0 来 创建 这 个 类 型 的 新 对 象 : 


//: generics/InstantiateGenericType.java 
import static net.mindview.util.Print.*; 


class ClassAsFactory<T> { 
Tx: 
public ClassAsFactory(Class<T> kind) { 
try { 
. X = kind. newInstance(); 
} catch(Exception e) { 
throw new RuntimeException(e); 
} 
$ 
} 


class Employee {} 


public class InstantiateGenericType { 
public static void main(String[] args) { 
ClassAsFactory<Employee> fe = 
new ClassAsFactory<Employee>(Employee.class); 
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print("ClassAsFactory<Employee> succeeded"); 
try { 
ClassAsFactory<Integer> fi = 
new ClassAsFactory<Integer> (Integer .class) ; 
} catch(Exception e) { 
print(*ClassAsFactory<Integer> failed"); 
} 
} 
} /* Output: 
ClassAsFactory<Employee> succeeded 
ClassAsFactory<Integer> failed 
“~ 


这 可 以 编译 ， 但 是 会 因 ClassAsFactory<Integer> 而 失败 ， 因 为 Integer 没 有 任何 默认 的 构造 
器 。 因 为 这 个 错误 不 是 在 编译 期 捕获 的 ， 所 以 Sun 的 伙计 们 对 这 种 方式 并 不 赞成 ， 他 们 建议 使 用 
显 式 的 工厂 ， 并 将 限制 其 类 型 ， 使 得 只 能 接受 实现 了 这 个 工厂 的 类 : 


//: generics/FactoryConstraint. java : 


interface FactoryI<T> { 
T create(); 
) 


class Foo2<T> { 
private T x; 
public <F extends FactoryI<T>> Foo2(F factory) { 
x = factory.create(); 


} 
Hs 
} ) 


class IntegerFactory implements FactoryI<Integer> { 
public Integer create() { 
1665. return new Integer (0); 
} 
} 


class Widget { 
public static class Factory implements FactoryI<Widget> { 
public Widget create() { 
return new Widget(); 
} 
} 
} 


public class FactoryConstraint { 
public static void main(String() args) { 
new Foo2<Integer>(new IntegerFactory()); 
new Foo2<Widget>(new Widget.Factory()); 














} 
} Mi~ 


注意 ， 这 确实 只 是 传递 Class<T> 的 一 种 变 体 。 两 种 方式 都 传递 了 工厂 对 象 ，Class<T> 碰 巧 
是 内 建 的 工厂 对 象 ， 而 上 面 的 方式 创建 了 一 个 显 式 的 工厂 对 象 ， 但 是 你 却 获得 了 编译 期 检查 。 

另 一 种 方式 是 模版 方法 设计 模式 。 在 下 面 的 示例 中 ，get0 是 模版 方法 ， 而 create0 是 在 子 类 
中 定义 的 、 用 来 产生 子 类 类 型 的 对 象 : 


//: generics/CreatorGeneric. java 


abstract class GenericWithCreate<T> { 
final T element; 
GenericWithCreate() { element = create(); } 
abstract T create(); 

j } 
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class X {} 


class Creator extends GenericWithCreate<X> { 
X create() { return new X(); } 
void fO { 
System. out.print1n(element.getCiass().getSimpleName()); 
} 
} 
public class CreatorGeneric { 
public static void main(String[] args) { 
Creator c = new Creator(); 
c.f0; 


} 

} /* Output: 

x 

Whe 

练习 22: (6) 使 用 类 型 标签 与 反射 来 创建 一 个 方法 ， 它 将 使 用 newInstance0 的 参数 版 本 来 创 
建 某 个 类 的 对 象 ， 这 个 类 拥有 一 个 具有 参数 的 构造 器 。 

练习 23，(]) 修改 FactoryConstraintjava， 使 得 create0 可 以 接受 一 个 参数 。 

练习 24，(3) 修改 练习 21， 使 得 工厂 对 象 是 由 一 个 Map 而 不 是 Class<?> 持 有 的 。 


15.8.2 泛 型 数组 

正如 你 在 Erased.java 中 所 见 ， 不 能 创建 泛 型 数组 。 一 般 的 解决 方案 是 在 任何 想 要 创建 泛 型 
数组 的 地 方 都 使 用 ArrayList: 

//: generics/ListOfGenerics.java 

import java.util.*; 


public class ListOfGenerics<T> { 

private List<T> array = new Arraylist<T>(); 

Public void add(T item) { array.add(item); } 

public T get(int index) { return array.get(index); } 
} Mi~ 


这 里 你 将 获得 数组 的 行为 ， 以 及 由 泛 型 提供 的 编译 期 的 类 型 安全 。 

有 了 时， 你 仍旧 希望 创建 泛 型 类 型 的 数组 (例如 ，ArrayList 内 部 使 用 的 是 数组 )。 有 趣 的 是 ， 
可 以 按照 编译 器 喜欢 的 方式 来 定义 一 个 引用 ， 例 如 : 

//: generics/ArrayOfGenericReference. java 


class Generic<T> {} 


public class ArrayOfGenericReference { 
static Generic<Integer>[] gia: 
dh 


编译 器 将 接受 这 个 程序 ， 而 不 会 产生 任何 警告 。 但 是 ， 永 远 都 不 能 创建 这 个 确切 类 型 的 数 
组 (包括 类 型 参数 )， 因 此 这 有 一 点 令 人 困惑 。 既 然 所 有 数组 无 论 它们 持 有 的 类 型 如 何 ， 都 具有 
相同 的 结构 〈 每 个 数组 槽 位 的 尺寸 和 数组 的 布局 ) ， 那 么 看 起 来 你 应 该 能 够 创建 一 个 Object 数组 ， 
并 将 其 转型 为 所 希望 的 数组 类 型 。 事 实 上 这 可 以 编译 ， 但 是 不 能 运行 ， 它 将 产生 ClassCase- 
Exception; 

//: generics/ArrayOfGeneric. java 


public class ArrayOfGeneric { 
static final int SIZE = 160; 
static Generic<Integer>[] gia; 
@SuppressWarnings("unchecked") 
Public static void main(Stringl] args) { 
// Compiles: produces ClassCastException: 
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//! gia = (Generic<Integer>[{])new Object (SIZE); 
// Runtime type is the raw (erased) type: 
gia = (Generic<Integer>[])new Generic [SIZE]: * 
System. out. printLn(gia.getClass().getSimpleName()) ; 
gia[0] = new Generic<Integer>(): 
//! gia[1] = new Object(); // Compile-time error 
// Discovers type mismatch at compile time: 
//! gia[2] = new Generic<Double>(); 
} 

} /* Output: 

Generic[] 

“I~ 


问题 在 于 数组 将 跟踪 它们 的 实际 类 型 ， 而 这 个 类 型 是 在 数组 被 创建 时 确定 的 ， 因 此 ， 即 使 
gia 已 经 被 转型 为 Generic<Integer>[]， 但 是 这 个 信息 只 存在 于 编译 期 (并 且 如 果 没 有 @Suppress 
Warnings 注 解 ， 你 将 得 到 有 关 这 个 转型 的 警告 )。 在 运行 时 ， 它 仍旧 是 Object 数组 ， 而 这 将 引发 
问题 。 成 功 创建 泛 型 数组 的 唯一 方式 就 是 创建 一 个 被 擦 除 类 型 的 新 数组 ， 然 后 对 其 转型 。 

让 我 们 看 一 个 更 复杂 的 示例 。 考 虚 一 个 简单 的 泛 型 数组 包装 器 : 

//: generics/GenericArray. java 

public class GenericArray<T> { 

private T[] array; 
@SuppressWarnings ("unchecked") 


public GenericArray(int sz) { 
array = (T[])new Object (sz); 


} 
public void put(int index, T item) { 
array(index] = item; 


} 
public T get(int index) { return array[index]; } 
// Method that exposes the underlying representation: 
public T[] rep() { return array; } 
public static void main(Stringl] args) { 

GenericArray<Integer> gai = 

new Gener icArray<Integer> (16) ; 

// This causes a ClassCastException: 

//! Integer[] ia = gai.rep(); 

1/ This 1s OK: 

Object] oa = gai-rep(); 


} 

} Ii~ 

与 前 面相 同 ， 我 们 并 不 能 声明 TU] array = new T[sz]， 因 此 我 们 创建 了 一 个 对 象 数组 ， 然 后 
将 其 转型 。 

rep() 方 法 将 返回 TD， 它 在 main() 中 将 用 于 gai， 因 此 应 该 是 Integer[]， 但 是 如 果 调 用 它 ， 并 
尝试 着 将 结果 作为 Integer[] 引 用 来 捕获 ， 就 会 得 到 ClassCastException， 这 还 是 因为 实际 的 运行 
时 类 型 是 ObjectD 。 

如 果 在 注释 掉 @SuppressWarnings 注 解 之 后 再 编译 GenericArray.java， 编 译 器 就 会 产生 
警告 : 


Note: GenericArray.java uses unchecked or unsafe operat ions. 
Note: Recompile with -Xlint:unchecked for details. 


在 这 种 情况 下 ， 我 们 将 只 获得 单个 的 警告 ， 并 且 相信 这 事 关 转型 。 但 是 如 果真 的 想 要 确定 
是 否 是 这 么 回 事 ， 就 应 该 用 -Xlint:unchecked 来 编译 : 
GenericArray.java:7: warning: [unchecked] unchecked cast 


found : java.lang.Object] 
required: TI] 
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array = (T[])new Object[sz]; 


1 warning 


这 确实 是 对 转型 的 抱怨 。 因 为 警告 会 变 得 令 人 迷惑 ， 所 以 一 旦 我 们 验证 某 个 特定 警告 是 可 
预期 的 ， 那 么 我 们 的 上 策 就 是 用 @Suppress Warnings 关 闭 它 。 通 过 这 种 方式 ， 当 警告 确实 出 现 
时 ， 我 们 就 可 以 真正 地 展开 对 它 的 调查 了 。 

因为 有 了 擦 除 ， 数 组 的 运行 时 类 型 就 只 能 是 Objett0D。 如 果 我 们 立即 将 其 转型 为 T[]， 那 么 
在 编译 期 该 数组 的 实际 类 型 就 将 丢失 ， 而 编译 器 可 能 会 错过 某 些 潜在 的 错误 检查 。 正 因为 这 样 ， 
最 好 是 在 集合 内 部 使 用 Object[]， 然 后 当 你 使 用 数组 元 素 时 ， 添 加 一 个 对 T 的 转型 。 让 我 们 看 看 
这 是 如 何 作用 于 GenericArrayjava 示 例 的 : 


//: generics/GenericArray2. java 


public class GenericArray2<T> { 
private Object[] array; 
public GenericArray2(int sz) { 
array = new Object[sz]; 


} 
public void put(int index, T item) { 
array[index] = item; 


@SuppressWarnings ("unchecked") 
public T get(int index) { return (T)array[index]; } 
@SuppressWarnings ("unchecked") 
public TE] rep() { 

return (T[])array; // Warning: unchecked cast 


} 
public static void main(String(] args) { 
GenericArray2<Integer> gai = 
new Gener icArray2<Integer>(10); 
for(int i = 6; 1 < 10; 1 ++) 
gai.put(i, 1); 
for(int i = 0; 1 < 10; 1 ++) 
System.out.print(gai.get(i) +" "); 
System.out.printin(); 
try { 
Integer[] ia = gai.rep(): 
} catch(Exception e) { System.out.printin(e); } 
} 
} /* Output: (Sample) 
6123456789 
java. lang.ClassCastException: [Ljava.lang.Object; cannot be 
cast to [Ljava.lang. Integer; 
Wham 


初 看 起 来 ， 这 好 像 没 多 大 变化 ， 只 是 转型 挪 了 地 方 。 如 果 没 有 @SuppressWarnings 注 解 ， 
你 仍旧 会 得 到 unchecked 警 告 。 但 是 ， 现 在 的 内 部 表示 是 Objeet[] 而 不 是 TU]。 当 getO 被 调用 时 ， 
它 将 对 象 转型 为 T， 这 实际 上 是 正确 的 类 型 ， 因 此 这 是 安全 的 。 然 而 ， 如 果 你 调用 rep0， 它 还 
是 尝试 着 将 Object 转型 为 TD， 这 仍旧 是 不 正确 的 ， 将 在 编译 期 产生 警告 ， 在 运行 时 产生 异常 。 
因此 ， 没 有 任何 方式 可 以 推翻 底层 的 数组 类 型 ， 它 只 能 是 ObjectD。 在 内 部 将 array 当 作 Object[] 
而 不 是 TD 处 理 的 优势 是 : 我 们 不 太 可 能 忘记 这 个 数组 的 运行 时 类 型 ， 从 而 意外 地 引入 缺陷 OR 
管 大 多 数 也 可 能 是 所 有 这 类 缺陷 都 可 以 在 运行 时 快速 地 探测 到 )。 

对 于 新 代码 ， 应 该 传递 一 个 类 型 标记 。 在 这 种 情况 下 ，GenericArray 看 起 来 会 像 下面 这 样 : 


//: generics/GenericArrayWithTypeToken. java 
import java.lang.reflect.*; 


public class GenericArraywWithTypeToken<T> { 








670| 














[672] 


386 RISE 





private T[] array; 

@SuppressWarnings ("unchecked") 

public GenericArraywWithTypeToken(Class<T> type, int sz) { 
array = (T[])Array.newInstance(type, sz); 


public void put(int index. T item) { 
array[index] = item: 
} 
public T get(int index) { return arraylindex]; } 
// Expose the underlying representation: 
public Tf] rep() { return array; } 
public static void main(String{] args) { 
GenericArraywithTypeToken<Integer> gai = 
new GenericArraywi thTypeToken<Integer>( 
Integer.class, 10); 
JJ This now works: 
Integer{] ia = gai-rep(): 


dha 
类 型 标记 Class<T> 被 传递 到 构造 器 中 ， 以 便 从 擦 除 中 恢复 ， 使 得 我 们 可 以 创建 需要 的 实际 
类 型 的 数组 ， 尽 管 从 转型 中 产生 的 警告 必须 用 @SuppressWarnings 压 制 住 。 一 旦 我 们 获得 了 实 
际 类 型 ， 就 可 以 返回 它 ， 并 获得 想 要 的 结果 ， 就 像 在 main0 中 看 到 的 那样 。 该 数组 的 运行 时 类 
型 是 确切 类 型 TD 。 . 
遗憾 的 是 ， 如 果 查 看 Java SE5 标 准 类 库 中 的 源 代 码 ， 你 就 会 看 到 从 Object 数组 到 参数 化 类 型 
的 转型 遍及 各 处 。 例 如 ， 下 面 是 经 过 整理 和 简化 之 后 的 从 Collection 中 复制 ArrayList 的 构造 器 : 


public ArrayList(Cottection c) { 
size = c.size(); 
elementData = (E[])new Object[size]; 
c. toArray(elementData) ; 


如 果 你 通读 ArrayListjava， 就 会 发 现 它 充满 了 这 种 转型 。 如 果 我 们 编译 它 ， 又 会 发 生 什 
AWE? 


Note: ArrayList. java uses unchecked or unsafe operations. 
Note: Recompile with -Xlint:unchecked for details. 


可 以 十 分 肯定 ， 标 准 类 库 会 产生 大 量 的 警告 。 如 果 你 曾经 用 过 C++， 特 别 是 ANSI C 之 前 的 
版 本 ， 你 就 会 记得 警告 的 特殊 效果 : 当 你 发 现 可 以 忽略 它们 时 ， 你 就 可 以 忽略 。 正 是 因为 这 个 
原因 ， 最 好 是 从 编译 器 中 不 要 发 出 任何 消息 ， 除 非 程序 员 必 须 对 其 进行 响应 。 

Neal Gafter (Java SE5 的 领导 开发 者 之 一 ) 在 他 的 博客 中指 出， 在 重 写 Java 类 库 时 ， 他 十 
分 懒散 ， 而 我 们 不 应 该 像 他 那样 。Neal 还 指出 ， 在 不 破坏 现 有 接口 的 情况 下 ， 他 将 无 法 修改 某 
些 Java 类 库 代码 。 因 此 ， 即 使 在 Java 类 库 源 代码 中 出 现 了 某 些 惯用 法 ， 也 不 能 表示 这 就 是 正确 的 
解决 之 道 。 当 查看 类 库 代码 时 ， 你 不 能 认为 它 就 是 应 该 在 自己 的 代码 中 遵循 的 示例 。 


15.9 边界 


本 章 前 面 简单 地 介绍 过 边界 。 边 界 使 得 你 可 以 在 用 于 泛 型 的 参数 类 型 上 设置 限制 条 件 。 尽 
管 这 使 得 你 可 以 强制 规定 泛 型 可 以 应 用 的 类 型 ， 但 是 其 潜在 的 一 个 更 重要 的 效果 是 你 可 以 按照 
自己 的 边界 类 型 来 调用 方法 。 

因为 擦 除 移 除了 类 型 信息 ， 所 以 ， 可 以 用 无 界 泛 型 参数 调用 的 方法 只 是 那些 可 以 用 Object 
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调用 的 方法 。 但 是 ， 如 果 能 够 将 这 个 参数 限制 为 某 个 类 型 子 集 ， 那 么 你 就 可 以 用 这 些 类 型 子 集 
来 调用 方法 。 为 了 执行 这 种 限制 ，Java 泛 型 重用 了 extends 关 键 字 。 对 你 来 说 有 一 点 很 重要 ， 即 
要 理解 extends 关 键 字 在 泛 型 边界 上 下 文 环境 中 和 在 普通 情况 下 所 具有 的 意义 是 完全 不 同 的 。 下 
面 的 示例 展示 了 边界 的 基本 要 素 : 


//: generics/BasicBounds. java 
interface HasColor { java.awt.Color getColor(); } 


class Colored<T extends HasColor> { 
T item; 
Colored(T item) { this.item = item; } 
T getItem() { return item; } 
// The bound allows you to call a method: 
java.awt.Color color() { return item.getColor(); 


} 








class Dimension { public int x. y, z; } 
// This won't work -- class must be first, then interfaces: 
// class ColoredDimension<T extends HasColor & Dimension> { 


// Multiple bounds: 
class ColoredDimension<T extends Dimension & HasColor> { 
T item; 
ColoredDimension(T item) { this.item = item; } 
T getItem() { return item; } 
java.awt.Color color() { return item.getColor(); } 
int getX() { return item.x; } 
int getY() { return item.y; } 
int getZ() { return item.z; } 
} 


interface Weight { int weight(); } 





// As with inheritance, you can have only one 

// concrete class but multiple interfaces: 

class Solid<T extends Dimension & HasColor & Weight> { rs 
T item; 
Solid(T item) { this.item = item; } 
T getItem() { return item; } 
java.ant.Color color() { return item.getCotor(): } 
int getX() { return item.x; } 
int getY() { return item.y; } 
int getZ() { return item.z; } 

bd int weight() { return item.weight(); } 
} 


class Bounded 

extends Dimension implements HasColor, Weight { 
public java.awt.Color getColor() { return null; } 
public int weight() { return @; } 

? 


public class BasicBounds { 
public static void main(String[] args) { 
Solid<Bounded> solid = 
new Solid<Bounded> (new Bounded()); 
solid.color(); 
solid.getY(): 
solid.weight(); 
} 
} Mi~ 


你 可 能 已 经 观察 到 了 ，BasicBounds.java 看 上 去 包含 可 以 通过 继承 消除 的 元 余 。 下 面 ， 可 以 
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看 到 如 何在 继承 的 每 个 层次 上 添加 边界 限制 : 


//: generics/InheritBounds. java 


Class HoldItem<T> { 
T item; 
Holditem(T item) { this.item = item; } 
T getItem() { return item; } 

} 


class Colored2<T extends HasColor> extends HoldItem<T> { 


Colored2(T item) { super(item); } 
java.awt.Color color() { return item.getColor(); } 


} 


class ColoredDimension2<T extends Dimension & HasColor> 
extends Colored2<T> { 

ColoredDimension2(T item) { super(item); } 

int getX() { return item.x; } 

int getY() { return item.y; } 

int getZ() { return item.z; } 
} 


class Solid2<T extends Dimension & HasColor & Weight> 
extends ColoredDimension2<T> { 

Solid2(T item) { super(item); } 

int weight() { return item.weight(); } 
)} 


public class InheritBounds { 
public static void main(String{) args) { 
Solid2<Bounded> solid? = 
new Solid2<Bounded>(new Bounded()); 
solid2.color(); 
solid2.getY(); 
solid2.weight(): 
} 
} Mi~ 


HoldItem 直 接 持 有 一 个 对 象 ， 因 此 这 种 行为 被 继承 到 了 Colored2 中 ， 它 也 要 求 其 参数 与 
HasColor 一 致 。ColoredDimension2 和 Solid2 进 一 步 扩展 了 这 个 层次 结构 ， 并 在 每 个 层次 上 都 添 
加 了 边界 。 现 在 这 些 方法 被 继承 ， 因 而 不 必 在 每 个 类 中 重复 。 

下 面 是 具有 更 多 层次 的 示例 : 


/1/: Benerics/EpicBattte.java 
// Demonstrating bounds in Java generics. 
import java.util.*; 











interface SuperPower {} 

interface XRayVision extends SuperPower { 
void seeThroughWalls(); 

) 

interface SuperHearing extends SuperPower { 
void hearSubtleNoises(): 

} 

interface SuperSmell extends SuperPower { 
void trackBySmell(); 

} 


class SuperHero<POWER extends SuperPower> { 
POWER power; 
SuperHero(POWER power) { this.power = power; } 
POWER getPower() { return power; } 

} 
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class SuperSleuth<POWER extends XRayVision> 

extends SuperHero<POWER> { 
SuperSleuth(POWER power) { super(power); } 
void see() { power.seeThroughWalls(); } 

J 


class CanineHero<POWER extends SuperHearing & SuperSmel1> 
extends SuperHero<POWER> { ' 
CanineHero(POWER power) { super (power); } 
void hear() { power.hearSubtleNoises(); } 
void smell() { power.trackBySmell(); } 
$ 


class SuperHearSmell implements SuperHearing, SuperSmell { 
public void hearSubtleNoises() {} 
public void trackBySmell() {} 

} 


class DogBoy extends CanineHero<SuperHearSmell> { 
DogBoy() { super(new SuperHearSmell()); } 


public class EpicBattle { 

// Bounds in generic methods: 

static <POWER extends SuperHearing> 

void useSuperHearing(SuperHero<POWER> hero) { 
hero. getPower () .hearSubtleNoises(); 

} 

Static <POWER extends SuperHearing & SuperSmell> 

Void superFind(SuperHero<POWER> hero) { 
hero. getPower () .hearSubtleNcises() ; 
hero.getPower().trackBySmel1(); 


} 
public static void main(String[] atgs) { 
DogBoy dogBoy = new DogBoy(); 
useSuperHear ing (dogBoy) ; 
super Find (dogBoy) ; 
// You can do this: 
List<? extends SuperHearing> audioBoys; 
// But you can't do this: 
11 Liste? extends SuperHearing & SuperSmel1> dogBoys; 


A 
) hm 
注意 ， 通 配 符 (我 们 下 面 将 要 学 习 ) 被 限制 为 单一 边界 。 
练习 25，(2) 创建 两 个 接口 和 一 个 实现 了 这 两 个 接口 的 类 。 创 建 两 个 泛 型 方法 ， 其 中 一 个 的 
参数 边界 被 限定 为 第 一 个 接口 ， 而 另 一 个 的 参数 边界 被 限定 为 第 二 个 接口 。 创 建 实现 了 这 两 个 
接口 的 类 的 实例 ， 并 展示 它 可 以 用 于 这 两 个 泛 型 方法 。 


15.10 通配符 


你 已 经 在 第 11 章 中 看 到 了 一 些 使 用 通配符 的 示例 一 在 泛 型 参数 表达 式 中 的 问号 ,在 第 14 章 
中 这 种 示例 更 多 。 本 节 将 更 深入 地 探讨 这 个 问题 。 

我 们 开始 入手 的 示例 要 展示 数组 的 一 种 特殊 行为 ， 可 以 向 导出 类 型 的 数组 赋予 基 类 型 的 数 
组 引用 ， 


//: generics/CovariantArrays.java 


class Fruit {} 
class Apple extends Fruit {} 
class Jonathan extends Apple {} 
class Orange extends Fruit {} 
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public class CovariantArrays { 
public static void main(Stringl] args) { 
Fruit[] fruit = new Apple[19] ; 
fruit[6] = new Apple(); // OK 
fruit[1] = new Jonathan(); // OK 
// Runtime type is Apple{], not Fruit[l] or Orange[]: 


try { 
7/ Compiler allows you to add Fruit: 
fruit[6] = new Fruit(); // ArrayStoreException 
} catch(Exception e) { System.out.printin(e); } 
try { 
// Compiler allows you to add Oranges: 
fruit[@] = new Orange(); // ArrayStoreException 
} catch(Exception e) { System.out.printin(e); } 
} 
} /* Output: 
java. lang.ArrayStoreException: Fruit 
java.lang.ArrayStoreException: Orange 
Wh 


main0 中 的 第 一 行 创 建 了 一 个 Apple 数 组 ， 并 将 其 赋值 给 一 个 Fruit 数 组 引用 。 这 是 有 意义 的 ， 
因为 Apple 也 是 一 种 Fruit， 因 此 Apple 数 组 应 该 也 是 一 个 Fruit 数 组 。 

但 是 ， 如 果实 际 的 数组 类 型 是 AppleD] ， 你 应 该 只 能 在 其 中 放置 Apple 或 Apple 的 子 类 型 ， 这 
在 编译 期 和 运行 时 都 可 以 工作 。 但 是 请 注意 ， 编 译 器 允许 你 将 Fruit 放 置 到 这 个 数组 中 ， 这 对 于 
编译 器 来 说 是 有 意义 的 ， 因 为 它 有 一 个 Fruit01 引 用 一 一 它 有 什么 理由 不 允许 将 Fruit 对 象 或 者 任 
何 从 Fruit 继 承 出 来 的 对 象 例如 Orange) ， 放 置 到 这 个 数组 中 呢 ? 因此 ， 在 编译 期 ， 这 是 允许 
的 。 但 是 ， 运 行 时 的 数组 机 制 知道 它 处 理 的 是 Apple[]， 因 此 会 在 向 数组 中 放置 异 构 类 型 时 抛 出 
异常 。 

实际 上 ， 向 上 转型 不 合适 用 在 这 里 。 你 真正 做 的 是 将 一 个 数组 赋值 给 另 一 个 数组 。 数 组 的 
行为 应 该 是 它 可 以 持 有 其 他 对 象 ， 这 里 只 是 因为 我 们 能 够 向 上 转型 而 已 ， 所 以 很 明显 ， 数 组 对 
象 可 以 保留 有 关 它 们 包含 的 对 象 类 型 的 规则 。 就 好 像 数组 对 它们 持 有 的 对 象 是 有 意识 的 ， 因 此 
在 编译 期 检查 和 运行 时 检查 之 间 ， 你 不 能 滥用 它们 。 

对 数组 的 这 种 赋值 并 不 是 那么 可 怕 ， 因 为 在 运行 时 可 以 发 现 你 已 经 插入 了 不 正确 的 类 型 。 
但 是 泛 型 的 主要 目标 之 一 是 将 这 种 错误 检测 移 人 到 编译 期 。 因 此 当 我 们 试图 使 用 泛 型 容器 来 代 
替 数 组 时 ， 会 发 生 什么 呢 ? 

//: generics/NonCovariantGenerics.java 

// {CompileTimeError} (Won't compile) 

import java.util.*; 

public class NonCovariantGenerics { 

// Compile Error: incompatible types: 
List<Fruit> flist = new ArrayList<Apple>(); 

H~ 

尽管 你 在 第 一 次 阅读 这 段 代码 时 会 认为 :“ 不 能 将 一 个 Apple 容 器 赋值 给 一 个 Fruit 容 器 "。 
别 忘 了 ， 泛 型 不 仅 和 容器 相关 正确 的 说 法 是 :“ 不 能 把 一 个 涉及 Apple 的 泛 型 赋 给 一 个 涉及 Fruit 
的 泛 型 "。 如 果 就 像 在 数组 的 情况 中 一 样 ， 编 译 器 对 代码 的 了 解 足 够 多 ， 可 以 确定 所 涉及 到 的 容 
器 ， 那 么 它 可 能 会 留 下 一 些 余地 。 但 是 它 不 知道 任何 有 关 这 方面 的 信息 ， 因 此 它 拒绝 向 上 转型 。 
然而 实际 上 这 根本 不 是 向 上 转型 一 -Apple 的 List 不 是 Fruit 的 List。Apple 的 List 将 持 有 Apple 和 
Apple 的 子 类 型 ， 而 Fruit 的 List 将 持 有 任何 类 型 的 Fruit， 诚 然 ， 这 包括 Apple 在 内 ， 但 是 它 不 是 
一 个 Apple 的 List， 它 仍旧 是 Fruit 的 List。Apple 的 List 在 类 型 上 不 等 价 于 Fruit 的 List， 即 使 
Apple 是 一 种 Fruit 类 型 。 
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真正 的 问题 是 我 们 在 谈论 容器 的 类 型 ， 而 不 是 容器 持 有 的 类 型 。 与 数组 不 同 ， 泛 型 没有 内 建 
的 协 变 类 型 。 这 是 因为 数组 在 语言 中 是 完全 定义 的 ， 因 此 可 以 内 建 了 编译 期 和 运行 时 的 检查 ， 但 是 
在 使 用 泛 型 时 ， 编 译 器 和 运行 时 系统 都 不 知道 你 想 用 类 型 做 些 什么 ， 以 及 应 该 采用 什么 样 的 规则 。 
但 是 ， 有 时 你 想 要 在 两 个 类 型 之 间 建 立 某 种 类 型 的 向 上 转型 关系 ， 这 正 是 通配符 所 允许 的 : 


//: generics/Gener icsAndCovariance. java 
import java.util.*: 


public class GenericsAndCovariance { 
public static void main(String[] args) { 
11 Wildcards allow covariance: 
Liste? extends Fruit> flist = new ArrayList<Apple>(): 
// Compile Error: can't add any type of object: 
77 flist.add(new Apple()); 
1/ flist.add(new Fruit()): 
/1 flist.add(new Object ()); 
flist.add(null); // Legal but uninteresting 
// We know that it returns at least Fruit: 
Fruit f = flist.get(@); (679} 





} 

} i~ 

flist 类 型 现在 是 List<? extends Fruit>， 你 可 以 将 其 读 作 “具有 任何 从 Fruit 继 承 的 类 型 的 列 
表 ”。 但是, 这 实际 上 并 不 意味 着 这 个 List 将 持 有 任何 类 型 的 Fruit。 通 配 符 引用 的 是 明确 的 类 型 
因此 它 意 味 着 “ 某 种 flist 引 用 没有 指定 的 具体 类 型 "。 因 此 这 个 被 赋值 的 List 必 须 持 有 诸如 Fruit 
或 Apple 这 样 的 某 种 指定 类 型 ， 但 是 为 了 向 上 转型 为 fist， 这 个 类 型 是 什么 并 没有 人 关心 。 

如 果 唯 一 的 限制 是 这 个 List 要 持 有 某 种 具体 的 Fruit 或 Fruit 的 子 类 型 ， 但 是 你 实际 上 并 不 关 
心 它 是 什么 ， 那 么 你 能 用 这 样 的 List 做 什么 呢 ? 如 果 不 知 道 List 持 有 什么 类 型 ， 那 么 你 怎样 才能 
安全 地 向 其 中 添加 对 象 呢 就 像 在 CovariantArraysjava 中 向 上 转型 数组 一 样 ， 你 不 能 ， 除 非 编 
译 器 而 不 是 运行 时 系统 可 以 阻止 这 种 操作 的 发 生 。 你 很 快 就 会 发 现 这 一 问题 。 

你 可 能 会 认为 ， 事 情 变 得 有 点 走 极端 了 ， 因 为 现在 你 甚至 不 能 向 刚刚 声明 过 将 持 有 Apple 对 
象 的 List 中 放置 一 个 Apple 对 象 了 。 是 的 ， 但 是 编译 器 并 不 知道 这 一 点 。List<? extends Fruit> 可 
以 合法 地 指向 一 个 List<Orange>。 一 旦 执行 这 种 类 型 的 向 上 转型 ， 你 就 将 丢失 掉 向 其 中 传递 任 
何 对 象 的 能 力 ， 甚 至 是 传递 Object 也 不 行 。 

另 一 方面 ， 如 果 你 调用 一 个 返回 Fruit 的 方法 ， 则 是 安全 的 ， 因 为 你 知道 在 这 个 List 中 的 任何 
对 象 至 少 具有 Fruit 类 型 ， 因 此 编译 器 将 允许 这 么 做 。 

练习 26，(2) 使 用 Number 和 Integer 证 明 数组 的 协 变性 。 

练习 27: (2) 使 用 Number 和 Integer， 然 后 引入 通配符 ， 以 此 展示 协 变性 对 List 不 起 作用 。 
15.10.1 编译 器 有 多 聪明 

现在 ， 你 可 能 会 猜想 自己 被 阻止 去 调用 任何 接受 参数 的 方法 ， 但 是 请 考虑 下 面 的 程序 : 


(/:, generics/CompiterIntelligence. java 
import java.util.*; 











public class CompilerIntelligence { [680] 
public static void main(String[] args) { 
List<? extends Fruit> flist = 
Arrays.asList(new Apple()); 
Apple a = (Apple)flist.get(9); // No warning 
flist.contains(new Apple()); // Argument is ‘Object’ 





flist. indexOf (new Apple()); // Argument is ‘Object’ 
} 
} Mi~ 
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你 可 以 看 到 ， 对 contains0 和 indexOf0 的 调用 ， 这 两 个 方法 都 接受 Apple 对 象 作为 参数 ， 而 
这 些 调用 都 可 以 正常 执行 。 这 是 否 意味 着 编译 器 实际 上 将 检查 代码 ， 以 查看 是 否 有 某 个 特定 的 
方法 修改 了 它 的 对 象 ? 

通过 查看 ArrayList 的 文档 ， 我 们 可 以 发 现 ， 编 译 器 并 没有 这 么 聪明 。 尽 管 add0 将 接受 一 个 
具有 泛 型 参数 类 型 的 参数 ， 但 是 contains0 和 indexOf0 将 接受 Object 类 型 的 参数 。 因 此 当 你 指定 
一 个 ArrayList<? extends Fruit> 时 ，add0 的 参数 就 变 成 了 “? Extends Fruit" 。 从 这 个 描述 中 ， 
编译 器 并 不 能 了 解 这 里 需要 Fruit 的 哪个 具体 子 类 型 ， 因 此 它 不 会 接受 任何 类 型 的 Fruit。 如 果 先 
将 Apple 向 上 转型 为 Fruit， 也 无 关 紧要 一 一 编译 器 将 直接 拒绝 对 参数 列表 中 涉及 通配符 的 方法 
(例如 add0) 的 调用 。 

在 使 用 contains0 和 indexOfO 时 ， 参 数 类 型 是 Object， 因 此 不 涉及 任何 通配符 ， 而 编译 器 也 
将 允许 这 个 调用 。 这 意味 着 将 由 泛 型 类 的 设计 者 来 决定 哪些 调用 是 “安全 的 "， 并 使 用 Object 类 
型 作为 其 参数 类 型 。 为 了 在 类 型 中 使 用 了 通配符 的 情况 下 禁止 这 类 调用 ， 我 们 需要 在 参数 列表 
中 使 用 类 型 参数 。 

可 以 在 一 个 非常 简单 的 Holder 类 中 看 到 这 一 点 : 


//: generics/Holder. java 


public class Holder<T> { 
private T value; 
public Holder() 4} 
public Holder(T val) { value 
public void set(T val) { valu 
public T get() { return value; 
Public boolean equals(Object obj) { 








return value.equals(obj): 
) 
public static void main(String{) args) { 
Holder<Apple> Apple = new Holder<Apple>(new Apple()); 
Apple d = Apple.get(): 
Apple. set (4); 
// Holder<Fruit> Fruit = Apple; // Cannot upcast 
Holder<? extends Fruit> fruit = Apple; // OK 
Fruit p = fruit.get(); 
d = (Apple) fruit.get(); // Returns ‘Object’ 
try { 
Orange c = (Orange) fruit.get(); // No warning 
} catch(Exception e) { System.out.printin(e); } 
// fruit.set(new Apple()); // Cannot call set() 
// fruit,set(new Fruit()): // Cannot call set() 
System.out.printin(fruit.equals(d)); // OK 
} 
} /* Output: (Sample) 
java. lang.ClassCastException: Apple cannot be cast to 
Orange 
true 
“Ii~ 


Holder 有 一 个 接受 T 类 型 对 象 的 setO 方 法 ， 一 个 get() 方 法 ， 以 及 一 个 接受 Object 对 象 的 
equals() 方 法 。 正 如 你 已 经 看 到 的 ， 如 果 创建 了 一 个 Holder<Apple>， 不 能 将 其 向 上 转型 为 
Holder<Fruit>， 但 是 可 以 将 其 向 上 转型 为 Holder<? extends Fruit>。 如 果 调 用 get0， 它 只 会 返 
回 一 个 Fruit 一 这 就 是 在 给 定 “ 任 何 扩展 自 Fruit 的 对 象 ”这 一 边界 之 后 ， 它 所 能 知道 的 一 切 了 。 
如 果 能 够 了 解 更 多 的 信息 ， 那 么 你 可 以 转型 到 某 种 具体 的 Fruit 类 型 ， 而 这 不 会 导致 任何 警告 ， 
但 是 你 存在 着 得 到 ClassCastException 的 风险 。set0 方 法 不 能 工作 于 Apple 或 Fruit， 因 为 set0 的 
参数 也 是 “? Extends Fruit"， 这 意味 着 它 可 以 是 任何 事物 ， 而 编译 器 无 法 验证 “任何 事物 ”的 
类 型 安全 性 。 
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但 是 ，equals0 方 法 工作 良好 ， 因 为 它 将 接受 Object 类 型 而 并 非 T 类 型 的 参数 。 因 此 ， 编 译 
器 只 关注 传递 进来 和 要 返回 的 对 象 类 型 ， 它 并 不 会 分 析 代 码 ， 以 查看 是 否 执行 了 任何 实际 的 写 
人 和 读 取 操作 。 
15.10.2 逆 变 

还 可 以 走 另 外 一 条 路 ， 即 使 用 超 类 型 通配符 。 这 里 ， 可 以 声明 通配符 是 由 某 个 特定 类 的 任 
何 基 类 来 界定 的 ， 方 法 是 指定 <? super MyClass>， 甚 至 或 者 使 用 类 型 参数 ，<? super T> (尽管 
你 不 能 对 泛 型 参数 给 出 一 个 超 类 型 边界 ， 即 不 能 声明 <T super MyClass>)。 这 使 得 你 可 以 安全 
地 传递 一 个 类 型 对 象 到 泛 型 类 型 中 。 因 此 ， 有 了 超 类 型 通配符 ， 就 可 以 向 Collection 写 人 了 : 


//: generics/SuperTypeWi ldcards. java 
import java.util.*; 


public class SuperTypeWildcards { 
static void writeTo(List<? super Apple> apples) { 
apples.add(new Apple()); 
apples. add(new Jonathan()); 
7/ apples.add(new Fruit()); // Error 


} 
} Mi~ 


参数 Apple 是 Apple 的 某 种 基 类 型 的 List， 这 样 你 就 知道 向 其 中 添加 Apple 或 Apple 的 子 类 型 是 
安全 的 。 但 是 ， 既 然 Apple 是 下 界 ， 那 么 你 可 以 知道 向 这 样 的 List 中 添加 Fruit 是 不 安全 的 ， 因 为 这 
将 使 这 个 List 散 开口 子 ， 从 而 可 以 向 其 中 添加 非 Apple 类 型 的 对 象 ， 而 这 是 违反 静态 类 型 安全 的 。 

因此 你 可 能 会 根据 如 何 能 够 向 一 个 泛 型 类 型 “ 写 入 ”( 传 递 给 一 个 方法 )， 以 及 如 何 能 够 从 
一 个 泛 型 类 型 中 “ 读 取 ”( 从 一 个 方法 中 返回 )， 来 着 手 思考 子 类 型 和 超 类 型 边界 。 

超 类 型 边界 放松 了 在 可 以 向 方法 传递 的 参数 上 所 作 的 限制 : 


//: generics/GenericWriting. java 
import java.util.*; 


public class GenericWriting { 
static <T> void writeExact(List<T> list, T item) { 
List.add (item) ; 
} 
static List<Apple> apples = new ArrayList<Apple>(); 
static List<Fruit> fruit = new ArrayList<Fruit>(); 
static void f1() { 
writeExact (apples, new Apple()); 
// writeExact (fruit, new Apple()); // Error: 
// Incompatible types: found Fruit, required Apple 
} 
static <T> void 
writeWithWildcard(List<? super T> list, T item) { 
List.add(item) ; 
} 


static void f2() { 
writeWithWildcard(apples, new Apple()); 
writeWithWildcard(fruit, new Apple()): 
} 
Public static void main(String[] args) { f1(); 20); } 
} Mi~ 


writeExact0) 方 法 使 用 了 一 个 确切 参数 类 型 (无 通配符 )。 在 f10 中 ， 可 以 看 到 这 工作 良 
好 一 一 只 要 你 只 向 List<Apple> 中 放置 Apple。 但 是 ，writeExact() 不 允许 将 Apple 放 置 到 
List<Fruit> 中 ， 即 使 知道 这 应 该 是 可 以 的 。 

在 writeWithWildcard0 中 ， 其 参数 现在 是 List<? super T>， 因 此 这 个 List 将 持 有 从 T 导 出 的 
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某 种 具体 类 型 ， 这 样 就 可 以 安全 地 将 一 个 T 类 型 的 对 象 或 者 从 T 导 出 的 任何 对 象 作 为 参数 传递 给 
List 的 方法 。 在 f20 中 可 以 看 到 这 一 点 ， 在 这 个 方法 中 我 们 仍旧 可 以 像 前 面 那 样 ， 将 Apple 放 置 到 
List<Apple> 中 ， 但 是 现在 我 们 还 可 以 如 你 所 期 望 的 那样 ， 将 Apple 放 置 到 List<Fruit> 中 。 

我 们 可 以 执行 下 面 这 个 相同 的 类 型 分 析 ， 作 为 对 协 变 和 通配符 的 一 个 复习 : 


//: generics/GenericReading. java 
import java.util.*; 


public class GenericReading { 
static <T> T readExact(List<T> list) { 
return list.get(0); 
} 
static List<Apple> apples = Arrays.asList(new Apple()); 
static List<Fruit> fruit = Arrays.asList(new Fruit()); 
// A static method adapts to each call: 
static void f1() { 
Apple a = readExact (apples); 
Fruit f = readExact(fruit); 
f = readExact (apples); 
} 
// If, however, you have a class, then its type is 
// established when the class is instantiated: 
static class Reader<T> { 
T readExact(List<T> list) { return list.get(@); } 


} 
static void f2() { 
Reader<Fruit> fruitReader = new Reader<Fruit>(); 
Fruit f = fruitReader.readExact (fruit); 
// Fruit a = fruitReader.readExact (apples); // Error: 
// readExact(List<Fruit>) cannot be 
// applied to (List<Apple>). 
$ } 
static class CovariantReader<T> { 
1 T readCovariant(List<? extends T> list) { 
return list.get(@); 


} 
static void 13() { 
CovariantReader<Fruit> fruitReader = 
new CovariantReader<Fruit>(); 
Fruit f = fruitReader.readCovariant (fruit); 
Fruit a = fruitReader.readCovariant (apples); 
} 
public static void main(String[] args) { 
F103 #20; £305 


} 
} Mi~ 


.与 前 面 一 样 ， 第 一 个 方法 readExact0 使 用 了 精确 的 类 型 。 因 此 如 果 使 用 这 个 没有 任何 通 配 
符 的 精确 类 型 ， 就 可 以 向 List 中 写 入 和 读 取 这 个 精确 类 型 。 另 外 ， 对 于 返回 值 ， 静 态 的 泛 型 方法 
readExact() 可 以 有 效 地 “适应 ”每 个 方法 调用 ， 并 能 够 从 List<Apple> 中 返回 一 个 Apple， 从 
List<Fruit> 中 返回 一 个 Fruit， 就 像 在 代 0 中 看 到 的 那样 。 因此， 如 果 可 以 摆脱 静态 泛 型 方法 ， 那 
么 当 只 是 读 取 时 ， 就 不 需要 协 变 类 型 了 。 

但 是 ， 如 果 有 一 个 泛 型 类 ， 那 么 当 你 创建 这 个 类 的 实例 时 ， 要 为 这 个 类 确定 参数 。 就 像 在 
f20 中 看 到 的 ，fruitReader 实 例 可 以 从 List<Fruit> 中 读 取 一 个 Fruit， 因 为 这 就 是 它 的 确切 类 型 。 
但 是 List<Apple> 还 应 该 产生 Fruit 对 象 ， 而 fruitReader 不 允许 这 么 做 。 

为 了 修正 这 个 问题 CovariantReader.readCovcariant0 方 法 将 接受 List<? extends T>, 因此 ， 
从 这 个 列表 中 读 取 一 个 T 是 安全 的 (你 知道 在 这 个 列表 中 的 所 有 对 象 至 少 是 一 个 T， 并 且 可 能 是 
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从 T 导 出 的 某 种 对 象 )。 在 人 0 中 ， 你 可 以 看 到 现在 可 以 从 List<Apple> 中 读 取 Fruit 了 。 

练习 28: (4) 创建 一 个 泛 型 类 Genericl<T>， 它 只 有 一 个 方法 ， 将 接受 一 个 T 类 型 的 参数 。 
创建 第 二 个 泛 型 类 Generic2<T>， 它 也 只 有 一 个 方法 ， 将 返回 类 型 T 的 参数 。 编 写 一 个 泛 型 方法 ， 
它 具 有 一 个 调用 第 一 个 泛 型 类 的 方法 的 逆 变 参数 。 编 写 第 二 个 泛 型 方法 ， 它 具有 一 个 调用 第 二 
个 泛 型 类 的 方法 的 协 变 参 数 。 使 用 typeinfo.pets 类 库 进行 测试 。 
15.10.3 无 界 通配符 

无 界 通配符 <?> 看 起 来 意味 着 “任何 事物 "， 因 此 使 用 无 界 通配符 好 像 等 价 于 使 用 原生 类 型 。 
事实 上 ， 编 译 器 初 看 起 来 是 支持 这 种 判断 的 : 


//: generics/UnboundedWildcards1. java 
import java.util.*; 


public class UnboundedWildcards1 { 

static List listl; 

static List<?> list2; 

static List<? extends Object> list3; 

static void assignl(List list) { 
Listl = list; 
list2 = list; 
Z1 Uist3 = list: // Warning: unchecked conversion 
// Found: List, Required: List<? extends Object> 


} 

static void assign2(List<?> list) { 
tisti = list; 
List? = list: 
Uist3 = list; 


} 

Static void assign3(List<? extends Object> list) { 
listl = list; 
list2 = list; 
Uist3 = list; 


} 

public static void main(String[] args) { 
assigni(new Arraylist()); 
assign2(new ArrayList()) ; 
// assign3(new ArrayList()); // Warning: 
// Unchecked conversion. Found: ArrayList 
// Required: List<? extends Object> 
assigni (new ArrayList<String>()); 
assign2(new ArrayList<String>()); 
assign3 (new ArrayList<String>()); 
// Both forms are acceptable as List<?>: 
List<?> wildList = new ArrayList(); 
wildList = new ArrayList<String>(); 
assignl(wildList); 
assign2(wildList); 
assign3(wildList); 


F 
} Mi~ 


有 很 多 情况 都 和 你 在 这 里 看 到 的 情况 类 似 ， 即 编译 器 很 少 关心 使 用 的 是 原生 类 型 还 是 <?>。 
在 这 些 情况 中 ，<?> 可 以 被 认为 是 一 种 装饰 ， 但 是 它 仍旧 是 很 有 价值 的 ， 因 为 ， 实 际 上 ， 它 是 在 
声明 :“ 我 是 想 用 Java 的 泛 型 来 编写 这 段 代码 ， 我 在 这 里 并 不 是 要 用 原生 类 型 ， 但 是 在 当前 这 种 
情况 下 ， 泛 型 参数 可 以 持 有 任何 类 型 。 

第 二 个 示例 展示 了 无 界 通 配 符 的 一 个 重要 应 用 。 当 你 在 处 理 多 个 泛 型 参数 时 ， 有 时 允许 一 
个 参数 可 以 是 任何 类 型 ， 同 时 为 其 他 参数 确定 某 种 特定 类 型 的 这 种 能 力 会 显得 很 重要 ; 


//: generics/UnboundedWitdcards2.java 
import java.util.*; 
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public class UnboundedWildcards2 { 
static Map mapl; 
static Map<?,?> map2! 
static Map<String,?> map3; 
static void assignl(Map map) { mapl = map; } 
static void assign2(Map<?,?> map) { map2 = maj 
static void assign3(Map<String,?2 map) { map3 
public static void main(String{] args) { 
assigni(new HashMap()): 
assign2(new HashMap()); 





api } 


// assign3(new HashMap()); // Warning: 
// Unchecked conversion. ‘Found: HashMap 
11 Required: Map<String, ?> 

assigni (new HashMap<String, Integer>()); 
assign2(new HashMap<Str ing, Integer>()): 
assign3(new HashMap<String, Integer>()): 





} 

} H~ 

但 是 ， 当 你 拥有 的 全 都 是 无 界 通配符 时 ， 就 像 在 Map<?,?> 中 看 到 的 那样 ， 编 译 器 看 起 来 就 
无 法 将 其 与 原生 Map 区 分 开 了 。 另 外 ，UnboundedWildcards.java 展 示 了 编译 器 处 理 List<?> 和 
List<? extends Object> 有 时 是 不 同 的 。 

令 人 困惑 的 是 ， 编 译 器 并 非 总 是 关注 像 List 和 List<?> 之 间 的 这 种 差异 ， 因 此 它们 看 起 来 就 
像 是 相同 的 事物 。 因 为 ， 事 实 上 ， 由 于 泛 型 参数 将 擦 除 到 它 的 第 一 个 边界 ， 因 此 List<?> 看 起 来 
等 价 于 List<Object>， 而 List 实 际 上 也 是 List<Object> 一 除非 这 些 语句 都 不 为 真 。List 实 际 上 表 
示 “ 持 有 任何 Object 类 型 的 原生 List"， 而 List<?> 表 示 “ 具 有 某 种 特定 类 型 的 非 原生 List， 只 是 
我 们 不 知道 那 种 类 型 是 什么 。” 

编译 器 何 时 才 会 关注 原生 类 型 和 涉及 无 界 通配符 的 类 型 之 间 的 差异 呢 ? 下 面 的 示例 使 用 了 
前 面 定义 的 Holder<T> 类 ， 它 包含 接受 Holder 作 为 参数 的 各 种 方法 ， 但 是 它们 具有 不 同 的 形式 ， 
作为 原生 类 型 ， 具 有 具体 的 类 型 参数 以 及 具有 无 界 通配符 参数 ， 

//; generics/Wildcards. java 

// Exploring the meaning of wildcards. 


public class Wildcards { 
// Raw argument: 
static void rawArgs(Holder holder, Object arg) { 
// holder.set (arg); // Warning: 
// Unchecked call to set(T) as a 
// member of the raw type Holder 
// holder.set (new Wildcards()); // Same warning 


// Can't do this; don't have any 'T': 
/1 T t = holder. get(); 


// OK, but type information has been lost: 
Object obj = holder.get(): 


// Similar to rawArgs(), but errors instead of warnings: 
static void unboundedArg(Holder<?> holder. Object arg) { 
// holder.set (arg); // Error: 
“a set(capture of ?) in Holder<capture of ?> 
// Cannot be applied to (Object) 
// holder.set(new Wildcards()); // Same error 


// Can't do this; don’t have any ‘T': 
// T t = holder.get(); 


// OK, but type information has been lost: 
Object obj = holder.get(): 
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static <T> T exactl(Holder<T> holder) { 
T t = holder.get(); 
return t; 


3 
static <T> T exact2(Holder<T> holder, T arg) { 


holder .set (arg); 
T t = holder.get(); 
return t; 
} 
static <T> 
T wildSubtype(Holder<? extends T> holder, T arg) { 
// holder.set(arg); // Error: 
// set(capture of ? extends T) in 
11 Wolder<capture of ? extends T> 
// cannot be applied to (T) 
Tt = holder.get(): 
return t; 
} 
static <T> 
void wildSupertype(Holder<? super T> holder, T arg) { 
holder. set (arg) ; 
1/ Tt = holder.get(); // Error: 
// Incompatible types: found Object, required T 


// OK, but type information has been lost: 
Object obj = holder.get(); 

} 

public static void main(String[] args) { 
Holder raw = new Holder<Long>(); 
11 Or: 
raw = new Holder (); 
Holder<Long> qualified = new Holder<Long>(); 
Holder<?> unbounded = new Holder<Long>(); 
Holder<? extends Long> bounded = new Holder<Long>(); 
Long Ing = 1L; 


rawArgs (raw, Ing): 
ranArgs (qualified, tng); 
rawArgs (unbounded, Ing): 


rawArgs (bounded, Ing): 


unboundedArg(raw. Ing); 
unboundedArg (qualified, Ing): 
unboundedArg(unbounded, Ing); 
unboundedArg (bounded, Ing); 


// Object r1 = exactl(raw); // Warnings: 

// Unchecked conversion from Holder to Holder<T> 
// Unchecked method invocation: exact1(Holder<T>) 
// is applied to (Holder) 

Long r2 = exact1 (qualified): 

Object r3 = exact1 (unbounded); // Must return Object 
Long r4 = exact1 (bounded) ; 


// Long r5 = exact2(raw, Ing); // Warnings: 

// Unchecked conversion from Holder to Holder<Long> 
// Unchecked method invocation: exact2(Holder<T>,T) 
// is applied to (Holder,Long) 

Long r6 = exact2(qualified, ing); 

// Long r7 = exact2(unbounded, 1ng); // Error: 

// exact2(Holder<T>,T) cannot be applied to 

1/ (Holder<capture of ?>,Long) 

71 Long r8 = exact2(bounded, 1ng); // Error: 

1/ exact2(Holder<T>,T) cannot be applied 

// to (Holder<capture of ? extends Long>,Long) 


11 Long r9 = 





ildSubtype (raw, Ing); // Warnings: 
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// Unchecked conversion from Holder 
/1/ to Holder<? extends Long> 

a Unchecked method invocation: 

1/ wildSubtype(Holder<? extends T>,T) is 
“a applied to (Holder,Long) 

Long r10 = wildSubtype(qualified, 1ng); 

// OK, but can only return Object: 

Object r11 = wildSubtype(unbounded, Ing); 
Long r12 = wildSubtype(bounded, 1ng): 


// wildSupertype(raw, Ing); // Warnings: 
// unchecked conversion from Holder 
// to Holder<? super Long> 
1/ Unchecked method invocation: 
1/ wildSupertype(Holder<? super T>,T) 
11 48 applied to (Holder ,Long) 
wildSupertype (qualified, Ing); 
// wildSupertype(unbounded, ing): // Error: 
1/ wildSupertype(Holder<? super T>,T) cannot be 
/1/ applied to (Holder<capture of ?>,Long) 
// wildSupertype(bounded, Ing); // Error: 
1/ wildSupertype(Holder<? super T>,T) cannot be 
// applied to (Holder<capture of ? extends Long>, Long) 
} 
} M~ 


在 rawArgs0 中 ， 编 译 器 知道 Holder 是 一 个 泛 型 类 型 ， 因 此 即使 它 在 这 里 被 表示 成 一 个 原生 
类 型 ， 编 译 器 仍旧 知道 向 set0 传 递 一 个 Object 是 不 安全 的 。 由 于 它 是 原生 类 型 ， 你 可 以 将 任何 类 
型 的 对 象 传递 给 set0， 而 这 个 对 象 将 被 向 上 转型 为 Object。 因 此 ， 无 论 何 时 ， 只 要 使 用 了 原生 类 
型 ， 都 会 放弃 编译 期 检查 。 对 get0 的 调用 说 明了 相同 的 问题 没有 任何 T 类 型 的 对 象 ， 因 此 结果 
只 能 是 一 个 Object。 

人 们 很 自然 地 会 开始 考虑 原生 Holder 与 Holder<?> 是 大 致 相同 的 事物 。 但 是 nboundedArg0 
强调 它们 是 不 同 的 一 一 它 揭 示 了 相同 的 问题 ， 但 是 它 将 这 些 问题 作为 错误 而 不 是 警告 报告 ， 因 为 
原生 Holder 将 持 有 任何 类 型 的 组 合 ， 而 Holder<?> 将 持 有 具有 某 种 具体 类 型 的 同 构 集合 ， 因 此 不 
能 只 是 向 其 中 传递 Object。 

在 exact10 和 exact20 中 ,你 可 以 看 到 使 用 了 确切 的 泛 型 参数 一 -没有 任何 通配符 。 你 将 看 到 ， 
exact20 与 exact10 具 有 不 同 的 限制 ， 因 为 它 有 额外 的 参数 。 

在 wildSubtype0 中 , 在 Holder 类 型 上 的 限制 被 放松 为 包括 持 有 任何 扩展 自 T 的 对 象 的 Holder。 
这 还 是 意味 着 如 果 T 是 Fruit， 那 么 holder 可 以 是 Holder<Apple>， 这 是 合法 的 。 为 了 防止 将 
Orange 放 置 到 Holder<Apple> 中 ， 对 set0 的 调用 (或 者 对 任何 接受 这 个 类 型 参数 为 参数 的 方法 的 
调用 ) 都 是 不 允许 的 。 但 是 ， 你 仍旧 知道 任何 来 自 Holder<? extends Fruit> 的 对 象 至 少 是 Fruit， 
因此 get0 (或 者 任何 将 产生 具有 这 个 类 型 参数 的 返回 值 的 方法 ) 都 是 允许 的 。 

wildSupertype0 展 示 了 超 类 型 通配符 ， 这 个 方法 展示 了 与 wildSubtype0 相 反 的 行为 ，holder 
可 以 是 持 有 任何 T 的 基 类 型 的 容器 。 因 此 ，setO 可 以 接受 T， 因 为 任何 可 以 工作 于 基 类 的 对 象 都 
可 以 多 态 地 作用 于 导出 类 (这 里 就 是 T)。 但 是 ， 尝 试 着 调用 get0 是 没有 用 的 ， 因 为 由 holder 持 
有 的 类 型 可 以 是 任何 超 类 型 ， 因 此 唯一 安全 的 类 型 就 是 Object。 

这 个 示例 还 展示 了 对 于 在 unbounded0 中 使 用 无 界 通配符 能 够 做 什么 不 能 做 什么 所 做 出 的 限 
制 。 对 于 迁移 兼容 性 ，rawArgs0 将 接受 所 有 Holder 的 不 同 变 体 ， 而 不 会 产生 警告 。unbounded- 
Arg( 方 法 也 可 以 接受 相同 的 所 有 类 型 ， 尽 管 如 前 所 述 ， 它 在 方法 体内 部 处 理 这 些 类 型 的 方式 并 
不 相同 。 

如 果 向 接受 “确切 ” 泛 型 类 型 (没有 通配符 ) 的 方法 传递 一 个 原生 Holder 引 用 ， 就 会 得 到 
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一 个 警告 ， 因 为 确切 的 参数 期 望 得 到 在 原生 类 型 中 并 不 存在 的 信息 。 如 果 向 exact10 传 递 一 个 无 
界 引用 ， 就 不 会 有 任何 可 以 确定 返回 类 型 的 类 型 信息 。 
可 以 看 到 ，exact20 具 有 最 多 的 限制 ， 因 为 它 希望 精确 地 得 到 一 个 Holder<T>， 以 及 一 个 具 
有 类 型 T 的 参数 ， 正 由 于 此 ， 它 将 产生 错误 或 警告 ， 除 非 提供 确切 的 参数 。 有 时 ， 这 样 做 很 好 ， 
但 是 如 果 它 过 于 受 限 ， 那 么 就 可 以 使 用 通配符 ， 这 取决 于 是 否 想 要 从 泛 型 参数 中 返回 类 型 确定 
的 返回 值 ( 就 像 在 wildSubtype0 中 看 到 的 那样 )， 或 者 是 否 想 要 向 泛 型 参数 传递 类 型 确定 的 参数 
(就 像 在 wildSupertype0 中 看 到 的 那样 )。 
因此 ， 使 用 确切 类 型 来 替代 通配符 类 型 的 好 处 是 ， 可 以 用 泛 型 参数 来 做 更 多 的 事 ， 但 是 使 
用 通配符 使 得 你 必须 接受 范围 更 宽 的 参数 化 类 型 作为 参数 。 因 此 ， 必 须 逐 个 情况 地 权衡 利 具 ， 
找到 更 适合 你 的 需求 的 方法 。 
15.10.4 捕获 转换 
有 一 种 情况 特别 需要 使 用 <?> 而 不 是 原生 类 型 。 如 果 向 一 个 使 用 <?> 的 方法 传递 原生 类 型 ， 
那么 对 编译 器 来 说 ， 可 能 会 推断 出 实际 的 类 型 参数 ， 使 得 这 个 方法 可 以 回转 并 调用 另 一 个 使 用 
这 个 确切 类 型 的 方法 。 下 面 的 示例 演示 了 这 种 技术 ， 它 被 称 为 捕获 转换 ， 因 为 未 指定 的 通配符 
类 型 被 捕获 ， 并 被 转换 为 确切 类 型 。 这 里 ， 有 关 警 告 的 注释 只 有 在 @SuppressWarnings 注 解 被 
移 除 之 后 才能 起 作用 ， 
//: generics/CaptureConversion. java 
public class CaptureConversion { 
static <T> void f1(Holder<T> holder) { 
T t = holder.get(); 
System.out.printin(t.getClass().getSimpleName()); 
buatte void f2(Holder<?> holder) { 
fi(holder); // Call with captured type 
Ssuppresswarnings ("unchecked") 
public static void main(String() args) { 
Holder raw = new Holder<Integer>(1); 
// fl(raw); // Produces warnings 
2(raw); // No warnings 
Holder rawBasic = new Holder(); 
rawBasic.set(new Object()); // Warning 
f2(rawBasic); // No warnings 
// Upcast to Holder<?>, still figures it out: 


Holder<?> wildcarded = new Holder<Double>(1.0); 
f2(wildcarded) ; 


} 
} /* Output: 
Integer 
Object 
Double 
“Ii~ 


人 10 中 的 类 型 参数 都 是 确切 的 ， 没 有 通配符 或 边界 。 在 f20 中 ，Holder 参 数 是 一 个 无 界 通 配 
符 ， 因 此 它 看 起 来 是 未 知 的 。 但 是 ， 在 f20 中 ， 身 0 被 调用 ， 而 们 0 需要 一 个 已 知 参数 。 这 里 所 发 
生 的 是 : 参数 类 型 在 调用 0 的 过 程 中 被 捕获 ， 因 此 它 可 以 在 对 人 40 的 调用 中 被 使 用 。 

你 可 能 想 知 道 ， 这 项 技术 是 否 可 以 用 于 写 入 ， 但 是 这 要 求 要 在 传递 Holder<?> 时 同时 传递 一 
个 具体 类 型 。 捕 获 转换 只 有 在 这 样 的 情况 下 可 以 工作 : 即 在 方法 内 部 ， 你 需要 使 用 确切 的 类 型 。 
注意 ， 不 能 从 20 中 返回 T， 因 为 T 对 于 f20 来 说 是 未 知 的 。 捕 获 转换 十 分 有 趣 ， 但 是 非常 受 限 。 

练习 29: (5) 创建 一 个 泛 型 方法 ， 它 接受 一 个 Holder<List<?>> 参 数 。 对 于 这 个 Holder 以 及 
这 个 List， 确 定 哪些 方法 是 可 以 调用 的 ， 哪 些 方法 是 不 可 以 调用 的 。 用 List<Holder<?>> 作 为 参 
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. 数 重复 这 个 练习 。 


15.11 问题 


本 节 将 阐述 在 使 用 Java 泛 型 时 会 出 现 的 各 类 问题 。 
15.11.1 任何 基本 类 型 都 不 能 作为 类 型 参数 

正如 本 章 早先 提 到 过 的 ， 你 将 在 Java 泛 型 中 发 现 的 限制 之 一 是 ， 不 能 将 基本 类 型 用 作 类 型 
参数 。 因 此 ， 不 能 创建 ArrayList<int> 之 类 的 东西 。 

解决 之 道 是 使 用 基本 类 型 的 包装 器 类 以 及 Java SE5 的 自动 包装 机 制 。 如 果 创 建 一 个 
ArrayList<Integer>， 并 将 基本 类 型 int 应 用 于 这 个 容器 ， 那 么 你 将 发 现 自 动 包装 机 制 将 自动 地 实 
现 int 到 Integer 的 双向 转换 一 因此 ， 这 几乎 就 像 是 有 一 个 ArrayList<int> 一 样 : 


//: generics/ListOfInt.java 

// Autoboxing compensates for the inability to use 
// primitives in generics. 

import java.util.*; 


public class ListOfInt { 
public static void main(String{] args) { 
List<Integer> li = new ArrayList<Integer>() ; 
for(int 1 = 0; i < 5; i++) 
Li.add(i): 
for(int 1 : 11) 
System.out.print(i +" "); 


} 如 Output: 

91234 

Wha 

注意 ， 自 动 包装 机 制 甚 至 允许 用 foreach 语 法 来 产生 int。 

通常 ， 这 种 解决 方案 工作 得 很 好 一 一 能 够 成 功 地 存储 和 读 取 int， 有 一 些 转换 碰巧 在 发 生 的 
同时 会 对 你 屏蔽 掉 。 但 是 ， 如 果 性 能 成 为 了 问题 ， 就 需要 使 用 专门 适 配 基本 类 型 的 容器 版 本 。 
Org.apache.commons.collections.primitives 就 是 一 种 开源 的 这 类 版 本 。 

下 面 是 另外 一 种 方式 ， 它 可 以 创建 持 有 Byte 的 Set: 

//: generics/ByteSet.java 

import java.util.*; 


public class ByteSet { 
Byte[] possibles = { 1,2,3.4,5,6,7,8.9 }; 
Set<Byte> mySet = 
new HashSet<Byte> (Arrays.asList(possibles)); 
// But you can't do this: 
// Set<Byte> mySet2 = new HashSet<Byte>( 
// Arrays. <Byte>asList(1,2,3,4,5,6,7,8,9)); 
} Mi~ 


注意 ， 自 动 包装 机 制 解决 了 一 些 问题 ， 但 并 不 是 解决 了 所 有 问题 。 下 面 的 示例 展示 了 一 个 
证 型 的 Generator 接 口 ， 它 指定 next( 方 法 返回 一 个 具有 其 参数 类 型 的 对 象 。FArray 类 包含 一 个 
泛 型 方法 ， 它 通过 使 用 生成 器 在 数组 中 填充 对 象 〔 这 使 得 类 泛 型 在 本 例 中 无 法 工作 ， 因 为 这 个 
方法 是 静态 的 ) 。Generator 实 现 来 自 第 16 章 ， 并 且 在 main0 中 ， 可 以 看 到 FArray.fill0 使 用 它 在 
数组 中 填充 对 象 : 


JL: generics/PrimitiveGenericTest.java 
import net.mindview.util.*; 


// Fill an array using a generator: 
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class FArray { 
public static <T> T[] fill(T[] a, Generator<T> gen) { 
for(int 1 = @; 1 < a length; i++) 
ali] = gen.next(); 
return a; 
} 
} 


public class PrimitiveGenericTest { 
public static void main(String{] args) { 
String[] strings = FArray.fill( 
new String(7]. new RandomGenerator .String(19)); 
for (String s : strings) 
System.out.println(s); 
Integer(] integers = FArray.fill( 
new Integer(7], new RandomGenerator. Integer()); 
for(int i: integers) 
System.out.printtn(i); 
// Autoboxing won't save you here. This won't compile: 
LA intl) b= 
// FArray.fill(new int{7], new RandIntGenerator()): 


} 
} /* Output: 
YNzbrnyGcF 
OWZnTcQrGs 
eGZHmJMROE 
suEcUOneOE 
dLsmwHLGEa 
hKcxrEqUCB 
bkInaMesbt 
7052 
6665 
2654 
3909 
5202 
2209 
5458 
Whim 


由 于 RandomGeneratorInteger 实 现 了 Generator<Integer>， 所 以 我 的 希望 是 自动 包装 机 制 
可 以 自动 地 将 next0 的 值 从 Integer 转 换 为 int。 但 是 ， 自 动 包装 机 制 不 能 应 用 于 数组 ， 因 此 这 无 
法 工作 。 

练习 30: (2) 为 每 一 种 基本 类 型 的 包装 器 类 型 都 创建 一 个 Holder， 并 展示 自动 包装 和 自动 拆 
包机 制 对 每 个 实例 的 set0 和 get0 方 法 都 起 作用 。 
15.11.2 实现 参数 化 接口 

一 个 类 不 能 实现 同一 个 泛 型 接口 的 两 种 变 体 ， 由 于 擦 除 的 原因 ， 这 两 个 变 体会 成 为 相同 的 
接口 。 下 面 是 产生 这 种 冲突 的 情况 : 


//: generics/MultipleInterfaceVariants. java 
// {CompileTimeError} (Won't compile) 


interface Payable<T> {} 


class Employee implements Payable<Employee> {} 


class Hourly extends Employee 
implements Payable<Hourly> {} ///:~ 


Hourly 不 能 编译 ， 因 为 擦 除 会 将 Payable<Employee> 和 Payable<Hourly> 简 化 为 相同 的 类 
了 Payable， 这 样 ， 上 面 的 代码 就 意味 着 在 重复 两 次 地 实现 相同 的 接口 。 十 分 有 趣 的 是 ， 如 果 从 
Payable 的 两 种 用 法 中 都 移 除 掉 泛 型 参数 (就 像 编译 器 在 擦 除 阶段 所 做 的 那样 》 这 段 代码 就 可 以 
编译 。 
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在 使 用 某 些 更 基本 的 Java 接 口 ， 例 如 Comparable<T> 时 ， 这 个 问题 可 能 会 变 得 十 分 令 人 恼 
火 ， 就 像 你 在 本 节 稍 后 就 会 看 到 的 那样 。 

练习 31: (1) 从 MultipleInterfaceVariants.java 中 移 除 所 有 泛 型 ， 并 修改 代码 ， 使 这 个 示例 
可 以 编译 。 
15.11.3 转型 和 警告 

使 用 带 有 泛 型 类 型 参数 的 转型 或 instanceof 不 会 有 任何 效果 。 下 面 的 容器 在 内 部 将 各 个 值 存 
储 为 Object， 并 在 获取 这 些 值 时 ， 再 将 它们 转型 回 T: 


//: generics/GenericCast.java 


class FixedSizeStack<T> { 
private int index = 0; 
private Object[] storage; 
public FixedSizeStack(int size) { 
storage = new Object [size]; 


} 
Public void push(T item) { storage[index++] = item; } 
@SuppressWarnings ("unchecked") 
public T pop() { return (T)storage[--index); } 
} 


public class GenericCast { 
public static final int SIZE = 16; 
public static void main(String{) args) { 
FixedSizeStack<String> strings = 
+ new FixedSizeStack<String> (SIZE); 
for(String s : "ABCDEF GHI J*.split(" ")) 
strings.push(s); 
for(int 1 = @; i < SIZE; 1++) { 
String s = strings.pop(); 
System.out.print(s +" "); 
y 


} 

} /* Output: 
JIHGFEDCBA 
Wha 


如 果 没有 @SuppressWarnings 注 解 ， 编 译 器 将 对 pop0 产 生 “unchecked cast” 和 警告。 由 于 控 
除 的 原因 ， 编 译 器 无 法 知道 这 个 转型 是 否 是 安全 的 ， 并 且 pop0 方 法 实际 上 并 没有 执行 任何 转型 。 
这 是 因为 ，T 被 擦 除 到 它 的 第 一 个 边界 ， 默 认 情况 下 是 Object， 因 此 pop0 实 际 上 只 是 将 Object 转 
型 为 Object。 

有 时， 泛 型 没有 消除 对 转型 的 需要 ， 这 就 会 由 编译 器 产生 警告 ， 而 这 个 警告 是 不 恰当 的 。 
例如 ， 


//: generics/NeedCasting. java 
import javasio.* 
import java.util 








public class NeedCasting { 
@SuppressWarnings("unchecked") 
public void f(String[] args) throws Exception { 
ObjectInputStream in = new ObjectInputStream( 
new FileInputStream(args[@])); 
List<Widget> shapes = (List<Widget>)in.readObject(); 


- Mh 


A 正如 你 将 在 下 一 章 学 到 的 那样 ，readObject0 无 法 知道 它 正在 读 取 的 是 什么 ， 因 此 它 返回 的 
是 必须 转型 的 对 象 。 但 是 当 注释 掉 @SuppressWarnings 注 解 ， 并 编译 这 个 程序 时 ， 就 会 得 到 下 
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面 的 警告: 


Note: NeedCasting.java uses unchecked or unsafe operations. 
_ Note: Recompile with -Xlint:unchecked for details. 


如 果 遵循 这 条 指示 ， 使 用 -Xlint: unchecked 来 重新 编译 ; 


NeedCasting. java:12: warning: [unchecked] unchecked cast 
found : java.lang.Object 
required: java.util.List<wWidget> 

List<Shape> shapes = (List<Widget>) in. readObject(); 


你 会 被 强制 要 求 转型 ， 但 是 又 被 告知 不 应 该 转型 。 为 了 解决 这 个 问题 ， 必 须 使 用 在 Java SES 
中 引入 的 新 的 转型 形式 ， 既 通过 泛 型 类 来 转型 : 


//: generics/ClassCasting. java 
import java. to.*; 
import java.util. 








public class ClassCasting { 
@SuppressWarnings ("unchecked") 
public void f(String[] args) throws Exception { 
ObjectInputStream in = new ObjectInputStream( 
new FileInputStream(args[9])); 
// Won't Compile: 
1/ List<Widget> lwl = 
//  List<Widget>.class.cast(in.readObject()); 
List<Widget> lw2 = List.class.cast(in.readObject()); 


} Wi 
但 是 ， 不 能 转型 到 实际 类 型 (List<Widget>) 。 也 就 是 说 ， 不 能 声明 
List<Widget>.class.cast(in.readObject()) 
甚至 当 你 添加 一 个 像 下 面 这 样 的 另 一 个 转型 时 : 
(List<Widget>)List.class.cast(in. readObject()) 
仍 介 会 得 到 一 个 警告 。 
练习 32， (1) 验证 在 GenericCast.java 中 的 FixedSizeStack 将 产生 异常 ， 如 果 试 图 超出 其 边界 
的 话 。 这 是 否 意味 着 边界 检查 代码 是 不 需要 的 ? 
练习 33，(3) 使 用 ArrayList 修 复 GenericCast.java。 
15.11.4 重 载 
下 面 的 程序 是 不 能 编译 的 ， 即 使 编译 它 是 一 种 合理 的 尝试 : 
//: generics/UseList.java 


// {CompileTimeError} (Won't compile) 
import java.util.*; 





public class UseList<W.T> { 
void f(List<T> v) {} 
void f(List<W> v) {} 

} Mi~ 


由 于 擦 除 的 原因 ， 重 载 方法 将 产生 相同 的 类 型 签名 。 

与 此 不 同 的 是 ， 当 被 控 除 的 参数 不 能 产生 唯一 的 参数 列表 时 ， 必 须 提供 明显 有 区 别 的 方法 名 : 
//: generics/UseList2.java 

import java.util.*; 


public class UseList2<w,T> { 
void f1(List<T> v) {} 
void f2(List<W> v) {} 

} Mi~ 
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幸运 的 是 ， 这 类 问题 可 以 由 编译 器 探测 到 。 
15.11.5 基 类 劫持 了 接口 
假设 你 有 一 个 Pet 类 ， 它 可 以 与 其 他 的 Pet 对 象 进 行 比较 (实现 了 Comparable 接 口 ): 


//: generics/ComparablePet. java 


public class ComparablePet 
implements Comparable<ComparablePet> { 

` public int compareTo(ComparablePet arg) { return @; } 
} Mi~ 


对 可 以 与 ComparablePet 的 子 类 比较 的 类 型 进行 窗 化 是 有 意义 的 ， 例 如 ， 一 个 Cat 对 象 就 只 
能 与 其 他 的 Cat 对 象 比较 : 


//: generics/HijackedInterface. java 
// {CompileTimeError} (Won't compile) 


Class Cat extends ComparablePet implements Comparable<Cat>{ 
// Error: Comparable cannot be inherited with 
// different arguments: <Cat> and <Pet> 
public int compareTo(Cat arg) { return 6; } 

} :~ 


遗憾 的 是 ， 这 不 能 工作 。 一 旦 为 Comparable 确 定 了 ComparablePet 参 数 ， 那 么 其 他 任何 实 
现 类 都 不 能 与 ComparablePet 之 外 的 任何 对 象 比较 : 

//: generics/RestrictedComparablePets. java 

class Hamster extends ComparablePet 

implements Comparable<ComparablePet> { 


public int compareTo(ComparablePet arg) { return 0; } 
} 


1/ Or just: 
class Gecko extends ComparablePet { 


Public int compareTo(ComparablePet arg) { return @; } 
} Mi~ 


Hamster 说 明 再 次 实现 ComparablePet 中 的 相同 接口 是 可 能 的 ， 只 要 它们 精确 地 相同 ， 包 括 
和 参数 类 型 在 内 。 但 是 ， 这 只 是 与 种 盖 基 类 中 的 方法 相同 ， 就 像 在 Gecko 中 看 到 的 那样 。 


15.12 自 限定 的 类 型 
在 Java 泛 型 中 ， 有 一 个 好 像 是 经 常 性 出 现 的 惯用 法 ， 它 相当 令 人 费解 : 


class SelfBounded<T extends SelfBounded<T>> { // ... 

这 就 像 两 面 镜子 彼此 照 向 对 方 所 引起 的 目眩 效果 一 样 ， 是 一 种 无 限 反射 。SelfBounded 类 接 
受 泛 型 参数 T， 而 T 由 一 个 边界 类 限定 ， 这 个 边界 就 是 拥有 T 作 为 其 参数 的 SelfBounded。 

当 你 首次 看 到 它 时 ， 很 难 去 解析 它 ， 它 强调 的 是 当 extends 关 键 字 用 于 边界 与 用 来 创建 子 类 
明显 是 不 同 的 。 
15.12.1 古怪 的 循环 泛 型 

为 了 理解 自 限定 类 型 的 含义 ， 我 们 从 这 个 惯用 法 的 一 个 简单 版 本 入手， 它 没 有 自 限定 的 


边界 。 


不 能 直接 继承 一 个 泛 型 参数 ， 但 是 ， 可 以 继承 在 其 自己 的 定义 中 使 用 这 个 泛 型 参数 的 类 。 


也 就 是 说 ， 可 以 声明 : 
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/1/: generics/CuriouslyRecurringGeneric. java 
class GenericType<T> {} 


public class CuriouslyRecurringGeneric 
extends GenericType<CuriouslyRecurringGeneric> {} ///:~ 


这 可 以 按照 jm CoplienZEC++ PFE MTMA RIRA, PAPE MRE 
(CRG), “古怪 的 循环 ”是 指 类 相当 古怪 地 出 现在 它 自己 的 基 类 中 这 一 事实 。 
为 了 理解 其 含义 ， 努 力 大 声 说 :“ 我 在 创建 一 个 新 类 ， 它 继承 自 一 个 泛 型 类 型 ， 这 个 泛 型 类 
型 接受 我 的 类 的 名 字 作为 其 参数 .” 当 给 出 导出 类 的 名 字 时 ， 这 个 泛 型 基 类 能 够 实现 什么 呢 ? 好 
吧 ，Java 中 的 泛 型 关 平 参数 和 返回 类 型 ， 因 此 它 能 够 产生 使 用 导出 类 作为 其 参数 和 返回 类 型 的 
基 类 。 它 还 能 将 导出 类 型 用 作 其 域 类 型 ， 甚 至 那些 将 被 擦 除 为 Object 的 类 型 。 下 面 是 表示 了 这 
种 情况 的 一 个 泛 型 类 : 
//: generics/BasicHolder.java 
public class BasicHolder<T> { 
T element; 
void set(T arg) { element = arg; } 
T get() { return element; } 


void tO { 
System. out.printin(element.getClass().getSimpleName()) ; 


} ite 
这 是 一 个 普通 的 泛 型 类 型 ， 它 的 一 些 方法 将 接受 和 产生 具有 其 参数 类 型 的 对 象 ， 还 有 一 个 
方法 将 在 其 存储 的 域 上 执行 操作 (尽管 只 是 在 这 个 域 上 执行 Object 操 作 )。 
我 们 可 以 在 一 个 古怪 的 循环 泛 型 中 使 用 BasicHolder: 
//: generics/CRGWithBasicHolder .java 
class Subtype extends BasicHolder<Subtype> {} 
public class CRGWithBasicHolder { 
public static void main(String[] args) { 
Subtype stl = new Subtype(), st2 = new Subtype(): 
stl.set(st2); 


Subtype st3 = st1.get(); 
st1.f0); 


} 
} /* Output: 
Subtype 
“~ 


注意 ， 这 里 有 些 东 西 很 重要 : 新 类 Subtype 接 受 的 参数 和 返回 的 值 具有 Subtype 类 型 而 不 仅 
仅 是 基 类 BasicHolder 类 型 。 这 就 是 CRG 的 本 质 : 基 类 用 导出 类 替代 其 参数 。 这 意味 着 泛 型 基 类 
变 成 了 一 种 其 所 有 导出 类 的 公共 功能 的 模版 ， 但 是 这 些 功能 对 于 其 所 有 参数 和 返回 值 ， 将 使 用 
导出 类 型 。 也 就 是 说 ， 在 所 产生 的 类 中 将 使 用 确切 类 型 而 不 是 基 类 型 。 因 此 ， 在 Subtype 中 ， 传 
递 给 set0 的 参数 和 从 get0 返 回 的 类 型 都 是 确切 的 Subtype。 
15.12.2 ARE 

BasicHolder 可 以 使 用 任何 类 型 作为 其 泛 型 参数 ， 就 像 下 面 看 到 的 那样 : 


/1/: generics/Unconstrained. java 


class Other {} 
class BasicOther extends BasicHolder<Other> {} 


public class Unconstrained { 


"> “g-a-i yo aes 
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public static void main(Stringl] args) { 
BasicOther b = new BasicOther(), b2 = new BasicOther(); 
b.set(new Other()); 
Other other = b.get(); 
b.fOs 


} 
} /* Output: 
Other 
Wx 
自 限定 将 采取 额外 的 步骤 ， 强 制 泛 型 当 作 其 自己 的 边界 参数 来 使 用 。 观 察 所 产生 的 类 可 以 
[6 因 《如 何 使 用 以 及 不 可 以 如 何 使 用 : 


11: generics/SelfBounding. java 


= 





class SelfBounded<T extends SelfBounded<T>> { 
T element; 
SelfBounded<T> set(T arg) { 
element = arg: 
return this; 


} 
T get() { return element; } 


class A extends SelfBounded<A> {} 
class B extends SelfBounded<A> {} // Also OK 


class C extends SelfBounded<C> { 
C setAndGet(C arg) { set(arg); return get(); } 


class D {} 

// Can't do this: 

// Class E extends SelfBounded<D> {} 

// Compile error: Type parameter D is not within its bound 


// Alas, you can do this, so you can’t force the idiom: 
class F extends SelfBounded {} 


public class SelfBounding { 
public static void main(String[] args) { 

A a = new AQ: 

a.set(new A()): 

a = a.set(new A()).get(); 

a= a.get(); 

C c = new CO; 

c = c.setAndGet (new C()); 


$ 

dh 

自 限定 所 做 的 ， 就 是 要 求 在 继承 关系 中 ， 像 下 面 这 样 使 用 这 个 类 : 

class A extends SelfBounded<A> {} 

[704 这 会 强制 要 求 将 正在 定义 的 类 当 作 参 数 传递 给 基 类 。 

自 限定 的 参数 有 何 意义 呢 ? 它 可 以 保证 类 型 参数 必须 与 正在 被 定义 的 类 相同 。 正 如 你 在 B 类 
的 定义 中 所 看 到 的 ， 还 可 以 从 使 用 了 另 一 个 SelfBounded 参 数 的 SelfBounded 中 导出 ， 尽 管 在 A 类 
看 到 的 用 法 看 起 来 是 主要 的 用 法 。 对 定义 E 的 尝试 说 明 不 能 使 用 不 是 SelfBounded 的 类 型 参数 。 

遗憾 的 是 ，F 可 以 编译 ， 不 会 有 任何 警告 ， 因 此 自 限定 惯用 法 不 是 可 强制 执行 的 。 如 果 它 确 
实 很 重要 ， 可 以 要 求 一 个 外 部 工具 来 确保 不 会 使 用 原生 类 型 来 替代 参数 化 类 型 。 

注意 ， 可 以 移 除 自 限定 这 个 限制 ， 这 样 所 有 的 类 仍旧 是 可 以 编译 的 ， 但 是 E 也 会 因此 而 变 得 
可 编译 ， 
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//: generics/NotSel fBounded. java 


public class NotSelfBounded<T> { 
T element; 
NotSelfBounded<T> set(T arg) { 
element = arg; 
return this; 


: 
T get() { return element; } 


class A2 extends NotSelfBounded<A2> {} 
class B2 extends NotSelfBounded<A2> {} 


class C2 extends NotSelfBounded<C2> { 
C2 setAndGet(C2 arg) { set(arg); return get(); } 
$ 


class D2 {} 
71 Now this is OK: 
class E2 extends NotSelfBounded<D2> {} ///:~ 


因此 很 明显 ， 自 限定 限制 只 能 强制 作用 于 继承 关系 。 如 果 使 用 自 限定 ， 就 应 该 了 解 这 个 类 
所 用 的 类 型 参数 将 与 使 用 这 个 参数 的 类 具有 相同 的 基 类 型 。 这 会 强制 要 求 使 用 这 个 类 的 每 个 人 
都 要 遵循 这 种 形式 。 

还 可 以 将 自 限定 用 于 泛 型 方法 : 


//: generics/SelfBoundingMethods.java 
public class SelfBoundingMethods { 
static <T extends SelfBounded<T>> T f(T arg) { 
return arg.set(arg).get(); 





} 
public static void main(String{] args) { 
A a = f(new AC); 


} 
YM 
这 可 以 防止 这 个 方法 被 应 用 于 除 上 述 形式 的 自 限 定 参数 之 外 的 任何 事物 上 。 
15.12.3 参数 协 变 
自 限定 类 型 的 价值 在 于 它们 可 以 产生 协 变 参数 类 型 一 一 方法 参数 类 型 会 随 子 类 而 变化 。 尽 
管 自 限定 类 型 还 可 以 产生 于 子 类 类 型 相同 的 返回 类 型 ， 但 是 这 并 不 十 分 重要 ， 因 为 协 变 返回 类 
型 是 在 Java SE5 中 引入 的 : 


/1/: generics/CovariantReturnTypes. java 


class Base {} 
Class Derived extends Base (} 


interface OrdinaryGetter { 
Base get(); 


interface DerivedGetter extends OrdinaryGetter { 
// Return type of overridden method is allowed to vary: 
Derived get); 


public class CovariantReturnTypes { 
void test(DerivedGetter d) { 
Derived d2 = d.getQ); 


} 
} Ii~ 
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[706] DerivedGetter 中 的 get( 方 法 覆盖 了 OrdinaryGetter 中 的 get0， 并 返回 了 一 个 从 Ordinary- 
Getter.get0 的 返回 类 型 中 导出 的 类 型 。 尽 管 这 是 完全 合乎 逻辑 的 事情 (导出 类 方法 应 该 能 够 返 
回 比 它 覆 盖 的 基 类 方法 更 具体 的 类 型 ) 但 是 这 在 早先 的 Java 版 本 中 是 不 合法 的 。 
自 限定 泛 型 事实 上 将 产生 确切 的 导出 类 型 作为 其 返回 值 ， 就 像 在 get0 中 所 看 到 的 一 样 : 


//: generics/GenericsAndReturnTypes.java 





interface GenericGetter<T extends GenericGetter<T>> { 
T get(); 
} 


interface Getter extends GenericGetter<Getter> {} 


public class GenericsAndReturnTypes { 
void test(Getter g) { 
Getter result = g-get(): 
GenericGetter gg = g-get(); // Also the base type 


} 
} Mi~ 


注意 ， 这 段 代码 不 能 编译 ， 除 非 是 使 用 囊括 了 协 变 返 回 类 型 的 Java SE5。 然 而 ， 在 非 泛 型 代 
码 中 ， 参 数 类 型 不 能 随 子 类 型 发 生变 化 ， 


//: generics/OrdinaryArguments. java 


class OrdinarySetter { 
void set(Base base) { 
System. out. println("OrdinarySetter.set (Base)") ; 
} 
i 


class DerivedSetter extends OrdinarySetter ( 
void set(Derived derived) { 
System.out.println("DerivedSetter .set(Derived)") ; 
} 
} 


publi class OrdinaryArguments { 
public static void main(String[] args) { 
Base base = new Base(); 
707| Derived derived = new Derived(); 
Derivedsetter ds = new DerivedSetter(); 
ds. set (derived) ; 
ds.set (base); // Compiles: overloaded, not overridden! 














} 
} /* Output: 
Derivedsetter.set (Derived) 
OrdinarySetter.set (Base) 
Wha 


set(derived)fiset(base) #2 2:05, HlLDerivedSetter.set()% A P X OrdinarySetter.set(), 
而 是 重 载 了 这 个 方法 。 从 输出 中 可 以 看 到 ， 在 DerivedSetter 中 有 两 个 方法 ， 因 此 基 类 版 本 仍旧 
是 可 用 的 ， 因 此 可 以 证 明 它 被 重 载 过 。 

但 是 ， 在 使 用 自 限 定 类 型 时 ， 在 导出 类 中 只 有 一 个 方法 ， 并 且 这 个 方法 接受 导出 类 型 而 不 
是 基 类 型 为 参数 : 


//: generics/SetfBoundingAndCovariantArguments.java 

interface SelfBoundSetter<T extends SelfBoundSetter<T>> { 
void set(T arg); 

} 


interface Setter extends SelfBoundSetter<Setter> {} 
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public class SelfBoundingAndCovariantArguments { 

void testA(Setter sl, Setter s2, SelfBoundSetter sbs) { 
s1.set(s2); 
// si.set(sbs); // Error: 
// set (Setter) in SelfBoundSetter<Setter> 
// cannot be applied to (SelfBoundSetter) 

} 

h~ 


编译 器 不 能 识别 将 基 类 型 当 作 参数 传递 给 set0 的 尝试 ， 因 为 没有 任何 方法 具有 这 样 的 签名 。 
实际 上 ， 这 个 参数 已 经 被 覆盖 。 

如 果 不 使 用 自 限定 类 型 ， 普 通 的 继承 机 制 就 会 介入 ， 而 你 将 能 够 重 载 ， 就 像 在 非 泛 型 的 情 
况 下 一 样 : 


/1/: generics/PlainGenericInheritance. java 
class GenericSetter<T> { // Not self-bounded 
void set(T arg){ 
System, out .println("GenericSetter.set(Base)"); 
} 
} 


class DerivedGS extends GenericSetter<Base> { 
void set(Derived derived) { 
System.out.println("DerivedGS.set (Derived) "); 
d 
d 


public class PlainGenericInheritance { 
public static void main(String() args) { 
Base base = new Base(): 
Derived derived = new Derived(); 
DerivedGS dgs = new DerivedGS(); 
dgs. set (derived) ; 
dgs.set (base); // Compiles: overloaded, not overridden! 


} 
} /* Output: 
DerivedGS.set (Derived) 
GenericSetter. set (Base) 
“Wh 


这 段 代码 在 模仿 OrdinaryArgument.java， 在 那个 示例 中 ，DerivedSetter 继 承 自 包 含 一 个 
set(Base) 的 OrdinarySetter。 而 这 里 ，DerivedGS 继 承 自 泛 型 创建 的 也 包含 有 一 个 set(Base) 的 
GenericSetter<Base>。 就 像 OrdinaryArgument.java 一 样 ， 你 可 以 从 输出 中 看 到 ，DerivedGS 包 
含 两 个 set0 的 重 载 版 本 。 如 果 不 使 用 自 限定 ， 将 重 载 参数 类 型 。 如 果 使 用 了 自 限定 ， 只 能 获得 
某 个 方法 的 一 个 版 本 ， 它 将 接受 确切 的 参数 类 型 。 

练习 34，(4) 创建 一 个 自 限定 的 泛 型 类 型 ， 它 包含 一 个 abstract 方 法 ， 这 个 方法 将 接受 一 个 
泛 型 类 型 参数 ， 并 产生 具有 这 个 泛 型 类 型 参数 的 返回 值 。 在 这 个 类 的 非 abstract 方 法 中 ， 调 用 这 
个 abstract 方 法 ， 并 返回 其 结果 。 继 承 这 个 自 限定 类 型 ， 并 测试 所 产生 的 类 。 


15.13 动态 类 型 安全 


因为 可 以 向 Java SE5 之 前 的 代码 传递 泛 型 容器 ， 所 以 旧式 代码 仍旧 有 可 能 会 破坏 你 的 容器 ， 
Java SE5 的 java.util.Collections 中 有 一 组 便利 工具 ， 可 以 解决 在 这 种 强 况 下 的 类 型 检查 问题 ， 它 
ME: 静态 方法 checkedCollection()、checkedList()、checkedMap()、checkedSet()、 
checkedSortedMap0 和 checkedSortedSet0。 这 些 方法 每 一 个 都 会 将 你 希望 动态 检查 的 容器 当 作 
第 一 个 参数 接受 ， 并 将 你 希望 强制 要 求 的 类 型 作为 第 二 个 参数 接受 。 
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受 检查 的 容器 在 你 试图 插入 类 型 不 正确 的 对 象 时 抛 出 ClassCastException， 这 与 泛 型 之 前 的 
(原生 ) 容器 形成 了 对 比 ， 对 于 后 者 来 说 ， 当 你 将 对 象 从 容器 中 取出 时 ， 才 会 通知 你 出 现 了 问题 。 
在 后 一 种 情况 中 ， 你 知道 存在 问题 ， 但 是 不 知道 罪魁 祸首 在 哪里 ， 如 果 使 用 受 检查 的 容器 ， 就 
可 以 发 现 谁 在 试图 插入 不 良 对 象 。 

让 我 们 用 受 检查 的 容器 来 看 看 “将 猫 插 入 到 狗 列表 中 ”这 个 问题 。 这 里 ，oldStyleMethod0 
表示 遗留 代码 ， 因 为 它 接受 的 是 原生 的 List， 而 @SuppressWarnings(“unchecked”) 注 解 对 于 
压制 所 产生 的 警告 是 必需 的 : 


//: generics/CheckedList. java 

// Using Collection.checkedList(). 
import typeinfo.pets.*; 

import java.util.*; 


public class CheckedList { 
@SuppressWarnings ("unchecked") 
Static void oldStyleMethod(List probablyDogs) { 
probablyDogs.add(new Cat()): 
} 
public static void main(String[] args) { 
List<Dog> dogsl = new ArrayList<Dog>(); 
oldStyleMethod(dogs1); // Quietly accepts a Cat 
List<Dog> dogs2 = Collections. checkedList( 
new ArrayList<Dog>(), Dog.class); 
try { 
oldStyleMethod(dogs2); // Throws an exception 
} catch(Exception e) { 
System.out.printin(e); 
} 
// Derived types work fine: 
List<Pet> pets = Collections.checkedList( 
new ArrayList<Pet>(), Pet.class); 
pets.add(new Dog()); 
pets.add(new Cat()): 
$ 
} /* Output: 
java. lang.ClassCastException: Attempt to insert class 
typeinfo.pets.Cat element into collection with element type 
class typeinfo.pets.Dog 
"Ii~ 


运行 这 个 程序 时 ， 你 会 发 现 插 入 一 个 Cat 对 于 dogs1 来 说 没有 任何 问题 ， 而 dogs2 立 即 会 在 这 
个 错误 类 型 的 插 和 人 操作 上 抛 出 一 个 异常 。 还 可 以 看 到 ， 将 导出 类 型 的 对 象 放置 到 将 要 检查 基 类 
型 的 受 检查 容器 中 是 没有 问题 的 。 

练习 35: (1) 修改 CheckedListjava， 使 其 使 用 本 章 中 定义 的 Coffee 类 。 


15.14 异常 


由 于 擦 除 的 原因 ， 将 泛 型 应 用 于 异常 是 非常 受 限 的 。cateh 语 句 不 能 捕获 泛 型 类 型 的 异常 ， 
因为 在 编译 期 和 运行 时 都 必须 知道 异常 的 确切 类 型 。 泛 型 类 也 不 能 直接 或 间接 继承 自 Throwable 
(这 将 进一步 阻止 你 去 定义 不 能 捕获 的 泛 型 异常 )。 

但 是 ， 类 型 参数 可 能 会 在 一 个 方法 的 throws 子 名 中 用 到 。 这 使 得 你 可 以 编写 随 检查 型 异常 
的 类 型 而 发 生变 化 的 泛 型 代码 : 


//: generics/ThrowGenericException.java 
import java.util.*; 


interface Processor<T,E extends Exception> { 
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void process(List<T> resultCollector) throws E; 
$ 


class ProcessRunner<T,E extends Exception> 
extends ArrayList<Processor<T,E>> { 
List<T> processAll() throws E { 
List<T> resultCollector = new ArrayList<T>(); 
for (Processor<T,£> processor : this) 
processor. process(resultCollector) ; 
return resultCollector; 
} 
} 


class Failurel extends Exception {} 


class Processorl implements Processor<String,Failurel> { 
static int count = 3; 
public void 
Process(List<String> resultCollector) throws Failurel { 
if(count-- > 1) 
FesultCollector.add("Hep!"); 
else 
resultCollector.add("Ho!"); 
if(count < 9) 
throw new Failurel(): 





} 
} 


class Failure2 extends Exception {} 


Class Processor2 implements Processor<Integer,Failure2> { 
static int count = 2; 
public void 
Process(List<Integer> resultCollector) throws Failure? { 
if(count-- == 6) 
resultCollector.add(47) ; 
else { 
resultCollector.add(11); 
} 
if(count < 9) 
throw new Failure2(); 
} 


} 


public class ThrowGenericException { 
public static void main(String{] args) { 
ProcessRunner<String,Failurel> runner 
new ProcessRunner<String,Failure1>( 
for(int i = 0; 1 < 3; 1++) 
runner.add(new Processor1()); 
try { 
System.out.printin(runner.processAll()); 
} catch(Failurel e) { 
System.out..printin(e); 
} 


ProcessRunner<Integer ,Failure2> runner2 = 
new ProcessRunner<Integer ,Failure2>(); 
for(int i = 0; 1 < 3; i++) 
runner2.add(new Processor2()): 
try { 
System. out.printin(runner2.processAll()); 
} catch(Failure2 e) { 
System. out.printin(e); 
} 
} 
} M~ 
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Processor 执 行 process()， 并 且 可 能 会 抛 出 具有 类 型 E 的 异常 。process() 的 结果 存储 在 
List<T> resultCollector 中 (这 被 称 为 收集 参数 ) 。ProcessRunner 有 一 个 processAll0 方 法 ， 它 将 
执行 所 持 有 的 每 个 Process 对 象 ， 并 返回 resultCollector。 

如 果 不 能 参数 化 所 抛 出 的 异常 ， 那 么 由 于 检查 型 异常 的 缘故 ， 将 不 能 编写 出 这 种 泛 化 的 代码 。 

练习 36: (2) 在 Processor 类 中 添加 第 二 个 参数 化 异常 ， 并 演示 这 些 异常 可 以 互相 独立 的 
变化 。 


15.15 混 型 


术语 混 型 随时 间 的 推移 好 像 拥 有 了 无 数 的 含义 ， 但 是 其 最 基本 的 概念 是 混合 多 个 类 的 能 力 ， 
以 产生 一 个 可 以 表示 混 型 中 所 有 类 型 的 类 。 这 往往 是 你 最 后 的 手段 ， 它 将 使 组 装 多 个 类 变 得 简 
单 易 行 。 

混 型 的 价值 之 一 是 它们 可 以 将 特性 和 行为 一 致 地 应 用 于 多 个 类 之 上 。 如 果 想 在 混 型 类 中 修 
改 某 些 东 西 ， 作 为 一 种 意外 的 好 处 ， 这 些 修改 将 会 应 用 于 混 型 所 应 用 的 所 有 类 型 之 上 。 正 由 于 
此 ， 混 型 有 一 点 面向 方面 编程 (AOP) 的 味道 ， 而 方面 经 常 被 建议 用 来 解决 混 型 问题 。 

15.15.1 C++ 中 的 混 型 

在 C++ 中 ， 使 用 多 重 继承 的 最 大 理由 ， 就 是 为 了 使 用 混 型 。 但 是 ， 对 于 混 型 来 说 ， 更 有 趣 、 
更 优雅 的 方式 是 使 用 参数 化 类 型 ， 因 为 混 型 就 是 继承 自 其 类 型 参数 的 类 。 在 C++ 中 ， 可 以 很 容 
易 的 创建 混 型 ， 因 为 C++ 能 够 记 住 其 模版 参数 的 类 型 。 

下 面 是 一 个 C++ 示例 ， 它 有 两 个 混 型 类 型 ,一 个 使 得 你 可 以 在 每 个 对 象 中 混 人 拥有 一 个 时 
间 惟 这 样 的 属性 ， 而 另 一 个 可 以 混和 人 一 个 序列 号 。 


//: generics/Mixins.cpp 
#include <string> 
#include <ctime> 
#include <iostream> 
using namespace std; 


template<class T> class TimeStamped : public T { 
long timeStamp; 

public: 

TimeStamped() { timeStamp = time(@); } 

long getStamp() { return timeStamp; } 

}; 


template<class T> class SerialNumbered : public T { 
long seriatNumber; 
static long counter; 
public: 
SerialNumbered() { serialNumber = counter++; } 
long getSerialNumber() { return serialNumber; } 
U 


// Define and initialize the static storage: 
template<class T> long SerialNumbered<T>::counter = 1; 


class Basic { 
string value; 

public: 
void set(string val) { value = val; } 
string get() { return value; } 


}; 


int main() { 
TimeStamped<SerialNumbered<Basic> > mixinl, mixin2: 
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mixinl.set("test string 1"); 
mixin2.set("test string 2" 
cout << mixinl.get() << " * << mixinl.getStamp() << 
" " << mixini.getSerialNumber() << endl; 
cout << mixin2.get() << " " << mixin2.getStamp() << 
”" << mixin2. getSerialNumber() << endl: 
} /* Output: (Sample) 
test string 1 1129840250 1 
test string 2 1129840250 2 
Wh~ 


在 main0 中 ，mixin1 和 mixin2 所 产生 的 类 型 拥有 所 混入 类 型 的 所 有 方法 。 可 以 将 混 型 看 作 是 
一 种 功能 ， 它 可 以 将 现 有 类 映射 到 新 的 子 类 上 。 注 意 ， 使 用 这 种 技术 来 创建 一 个 混 型 是 多 么 地 
轻而易举 。 基 本 上 ， 只 需要 声明 “这 就 是 我 想 要 的 "， 紧 跟着 它 就 发 生 了 : 

TimeStamped<SerialNumbered<Basic> > mixinl, mixin2; 

址 性 的 是 ，Java 泛 型 不 允许 这 样 。 控 除 会 忘记 基 类 类 型 ， 
EMER, 

15.15.2 与 接口 混合 
一 种 更 常见 的 推荐 解决 方案 是 使 用 接口 来 产生 混 型 效果 ， 就 像 下 面 这 样 : 


//: generics/Mixins. java 
import java.util.*; 








四 此 泛 型 类 不 能 直接 继承 自 一 个 


interface TimeStamped { long getStamp(); } 


class TimeStampedImp implements TimeStamped { 
private final long timestamp; 
public TimeStampedImp() { 
timeStamp = new Date().getTime(): 


} 
Public long getStamp() { return timeStamp; } 
} 





interface SerialNumbered { long getSerialNumber(); } 
class SeriatNumberedImp implements SerialNumbered { 

private static long counter = 1; 

private final long sertalNumber = counter++: 

public long getSerialNumber() { return serialNumber; } 
$ 


interface Basic { 
public void set(String val); 
public String get(); 

} 


class BasicImp implements Basic { 
private String value; 
Public void set(String val) { value = val: } 
public String get() { return value; } 

} 


class Mixin extends BasicImp 
implements TimeStamped, SerialNumbered { 
private TimeStamped timeStamp = new TimeStampedImp(); 
Private SerialNumbered serialNumber = 
new SerialNumberedImp() ; 
public long getStamp() { return timeStamp.getStamp(); } 
public long getSerialNumber() { 
return serialNumber.getSerialNumber() ; 
} 
} 


public class Mixins { 
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public static void main(String[] args) { 

Mixin mixin] = new Mixin(), mixin2 = new Mixin(); 

mixinl.set ("test string 1"); 

mixin2.set("test string 2"); 

System.out.printin(mixinl.get() + " "+ 
mixini.getStamp() + " ”+ mixinl.getSerialNumber()); 

System.out.printin(mixin2.get() + "+ 
mixin2.getStamp() + " " + mixin2.getSertalNumber()); 


) I. Output: (Sample) 

test string 1 1132437151359 1 

test string 2 1132437151359 2 

Mixin 类 基本 上 是 在 使 用 代理 ， 因 此 每 个 混 人 类 型 都 要 求 在 Mixin 中 有 一 个 相应 的 域 ， 而 你 
必须 在 Mixin 中 编写 所 有 必需 的 方法 ， 将 方法 调用 转发 给 恰当 的 对 象 。 这 个 示例 使 用 了 非常 简单 
的 类 ， 但 是 当 使 用 更 复杂 的 混 型 时 ， 代 码 数量 会 急速 增加 。。 

练习 37: (2) 向 Mixins.java 中 添加 一 个 新 的 混 型 类 ， 将 其 混入 到 Mixin 中 ， 并 展示 其 是 可 以 
工作 的 。 
15.15.3 使 用 装饰 器 模式 

当 你 观察 混 型 的 使 用 方式 时 ， 就 会 发 现 混 型 概念 好 像 与 装饰 器 设计 模式 关系 很 近 s 。 装 饰 
器 经 常用 于 满足 各 种 可 能 的 组 合 ， 而 直接 子 类 化 会 产生 过 多 的 类 ， 因 此 是 不 实际 的 。 

装饰 器 模式 使 用 分 层 对 象 来 动态 透明 地 向 单个 对 象 中 添加 责任 。 装 饰 器 指定 包装 在 最 初 的 
对 象 周围 的 所 有 对 象 都 具有 相同 的 基本 接口 。 某 些 事物 是 可 装饰 的 ， 可 以 通过 将 其 他 类 包装 在 
这 个 可 装饰 对 象 的 四 周 ， 来 将 功能 分 层 。 这 使 得 对 装饰 器 的 使 用 是 透明 的 一 一 无 论 对 象 是 否 被 
装饰 ， 你 都 拥有 一 个 可 以 向 对 象 发 送 的 公共 消息 集 。 装 饰 类 也 可 以 添加 新 方法 ， 但 是 正如 你 所 
见 ， 这 将 是 受 限 的 。 

装饰 器 是 通过 使 用 组 合 和 形式 化 结构 (可 装饰 物 /装饰 器 层次 结构 ) 来 实现 的 ， 而 混 型 是 基 
于 继承 的 。 因 此 可 以 将 基于 参数 化 类 型 的 混 型 当 作 是 一 种 泛 型 装饰 器 机 制 ， 这 种 机 制 不 需要 装 
饰 器 设计 模式 的 继承 结构 。 

前 面 的 示例 可 以 被 改写 为 使 用 装饰 器 : 


//: generics/decorator/Decoration. java 
package generics.decorator; 
import java.util.*; 


class Basic { 


private String value; 
public void set(String val) { value = val; } 
public String get() { return value; } 

} 


class Decorator extends Basic { 
protected Basic basic; 
public Decorator(Basic basic) { this.basic = basic; } 
public void set(String val) { basic.set(val); } 
public String get() { return basic.get(); } 

, 


class TimeStamped extends Decorator { 
private final long timeStamp: 


O 注意 ， 某 些 编程 环境 ， 例 如 Eclipse 和 Intelli Idea， 可 以 自动 地 生成 代理 代码 。 
© 模式 是 《Thinking in Patterns(with Java)》 的 主题 ， 可 以 在 www.MindViewnet 中 找到 这 本 书 。 还 可 以 查看 Erich 
Gamma 等 人 撰写 的 《Design Pattems) (Addison-Wesley, 1995)。《 设 计 模式 (双语 版 )》 已 由 机 械 工业 出 版 社 出 版 。 
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public TimeStamped(Basic basic) { 
super (basic) ; 
timeStamp = new Date().getTime(); 

} 

public long getStamp() { return timeStamp; } 


class SerialNumbered extends Decorator { 
private static long counter = 1; 
private final long serialNumber = counter++; 
public SerialNumbered(Basic basic) { super (basic); } 
public long getSerialNumber() { return serialNumber; } 
} 


public class Decoration { 
public static void main(String[] args) { 
TimeStamped t = new TimeStamped(new Basic()); 
TimeStamped t2 = new TimeStamped( 
new SerialNumbered(new Basic())); 
//! t2.getSerialNumber(); // Not available 
SerialNumbered s = new SerialNumbered(new Basic()); 
SerialNumbered s2 = new SerialNumbered( 
new TimeStamped(new Basic())); 
//! s2.getStamp(); // Not available 
$ Vis 
产生 自 泛 型 的 类 包含 所 有 感 兴趣 的 方法 ， 但 是 由 使 用 装饰 器 所 产生 的 对 象 类 型 是 最 后 被 装 
饰 的 类 型 。 也 就 是 说 ， 尽 管 可 以 添加 多 个 层 ， 但 是 最 后 一 层 才 是 实际 的 类型 ， 因 此 只 有 最 后 一 
层 的 方法 是 可 视 的 ， 而 混 型 的 类 型 是 所 有 被 混合 到 一 起 的 类 型 。 因 此 对 于 装饰 器 来 说 ， 其 明显 
的 缺陷 是 它 只 能 有 效 地 工作 于 装饰 中 的 一 层 (最 后 一 层 )， 而 混 型 方法 显然 会 更 自然 一 些 。 因 此 ， 
装饰 器 只 是 对 由 混 型 提出 的 问题 的 一 种 局 限 的 解决 方案 。 
练习 38，(4) 从 基本 的 咖啡 人手 ， 创 建 一 个 简单 的 装饰 器 系统 ， 然 后 提供 可 以 到 合 人 牛奶 、 
泡沫 、 巧 克 力 、 焦 糖 和 生 奶油 的 装饰 器 。 


15.15.4 与 动态 代理 混合 

可 以 使 用 动态 代理 来 创建 一 种 比 装饰 器 更 贴近 混 型 模型 的 机 制 (查看 第 14 章 中 关于 Java 的 
动态 代理 如 何 工作 的 解释 ) 。 通 过 使 用 动态 代理 ， 所 产生 的 类 的 动态 类 型 将 会 是 已 经 混 人 的 组 合 
类 型 。 

由 于 动态 代理 的 限制 ， 每 个 被 混入 的 类 都 必须 是 某 个 接口 的 实现 : 


//: gener ics/DynamicProxyMixin. java 
import java. lang. reflect.*; 
import java.util.*; 
import net.mindview.util.*; 
import static net.mindview.util.Tuple.*; 





class MixinProxy implements InvocationHandler { 
Map<String,Object> delegatesByMethod; 
public MixinProxy(TwoTuple<Object,Class<?>>... pairs) { 
delegatesByMethod = new HashMap<String,Object>(); 
for (TwoTuple<Object ,Class<?>> pair : pairs) { 
for(Method method : pair.second.getMethods()) { 
String methodName = method.getName(); 
// The first interface in the map 
// implements the method. 
if (!delegatesByMethod.containsKey (methodName) ) 
delegatesByMethod. put (methodName. pair. first): 





} 
} 
F 








718] 














416 FISH 





public Object invoke(Object proxy, Method method, 
Object{} args) throws Throwable { 
String methodName = method. getName(); 
Object delegate = delegatesByMethod.get (methodName) ; 
return method. invoke (delegate, args); 


} 
@SuppressWarnings ("unchecked") 
public static Object newInstance(TwoTuple... pairs) { 
Class{] interfaces = new Class[pairs. length] ; 
for(int i = 6; i < pairs.length; i++) { 
interfaces[i} = (Class)pairs[i].second; 


} 
ClassLoader cl = 
pairs(0].first.getClass() .getClassLoader(); 
return Proxy .newProxyInstance( 
cl, interfaces, new MixinProxy(pairs)); 
} 
} 


public class DynamicProxyMixin { 
public static void main(String[] args) { 
Object mixin = HixinProxy.newInstance( 
tuple(new BasicImp(), Basic.class), 
tuple(new TimeStampedImp(), TimeStamped.class). 
tuple(new SerialNumberedimp() .SerialNumbered.class)) ; 
Basic b = (Basic)mixin; 
TimeStamped t = (TimeStamped) mixin; 
SerialNumbered s = (SerialNumbered)mixin; 
b.set ("Hello"); 
System. out.printin(b.get()); 
System. out.printin(t.getStamp()); 
System. out.printin(s.getSerialNumber()); 


} h Output: (Sample) 

Hello 

1132519137015 

ain 

因为 只 有 动态 类 型 而 不 是 非 静 态 类 型 才 包含 所 有 的 混 人 类 型 ， 因 此 这 仍旧 不 如 C++ 的 方式 
好 ， 因 为 可 以 在 具有 这 些 类 型 的 对 象 上 调用 方法 之 前 ， 你 被 强制 要 求 必须 先 将 这 些 对 象 向 下 转 
型 到 恰当 的 类 型 。 但 是 ， 它 明显 地 更 接近 于 真正 的 混 型 。 

为 了 让 Java 支 持 混 型 ， 人 们 已 经 做 了 大 量 的 工作 朝 着 这 个 目标 努力 ， 包 括 创 建 了 至 少 一 种 
附加 语言 (Jam 语 言 )， 它 是 专门 用 来 支持 混 型 的 。 

练习 39， (1) 向 DynamicProxyMixin.java 中 添加 一 个 新 的 混 型 类 Colored， 将 其 混入 mixin， 
并 展示 它 可 以 工作 。 


15.16 潜在 类 型 机 制 


在 本 章 的 开头 介绍 过 这 样 的 思想 ， 即 要 编写 能 够 尽 可 能 广泛 地 应 用 的 代码 。 为 了 实现 这 一 
点 ， 我 们 需要 各 种 途径 来 放松 对 我 们 的 代码 将 要 作用 的 类 型 所 作 的 限制 ， 同 时 不 丢失 静态 类 型 
检查 的 好 处 。 然 后 ， 我 们 就 可 以 编写 出 无 需 修 改 就 可 以 应 用 于 更 多 情况 的 代码 ， 即 更 加 “ 泛 化 ” 
的 代码 。 

Java 泛 型 看 起 来 是 向 这 一 方向 迈进 了 一 步 。 当 你 在 编写 或 使 用 只 是 持 有 对 象 的 泛 型 时 ， 这 
些 代 码 将 可 以 工作 于 任何 类 型 (除了 基本 类 型 ， 尽 管 正如 你 所 见 到 的 ， 自 动 包装 机 制 可 以 克 
服 这 一 点 )。 或 者 ， 换 个 角度 讲 ,，“ 持 有 器 ” 泛 型 能 够 声明 :“ 我 不 关心 你 是 什么 类 型 "。 如 果 
代码 不 关心 它 将 要 作用 的 类 型 ， 那 么 这 种 代码 就 可 以 真正 地 应 用 于 任何 地 方 ， 并 因此 而 相当 
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“ 泛 化 ”。 

还 是 正如 你 所 见 到 的 ， 当 要 在 泛 型 类 型 上 执行 操作 ( 即 调用 Object 方 法 之 前 的 操作 ) 时 
就 会 产生 问题 ， 因 为 擦 除 要 求 指定 可 能 会 用 到 的 泛 型 类 型 的 边界 ， 以 安全 地 调用 代码 中 的 泛 型 
对 象 上 的 具体 方法 。 这 是 对 “ 泛 化 ”概念 的 一 种 明显 的 限制 ， 因 为 必须 限制 你 的 泛 型 类 型 , 使 
它们 继承 自 特 定 的 类 ， 或 者 实现 特定 的 接口 。 在 某 些 情况 下 ， 你 最 终 可 能 会 使 用 普通 类 或 普通 
接口 ， 因 为 限定 边界 的 泛 型 可 能 会 和 指定 类 或 接口 没有 任何 区 别 。 

某 些 编程 语言 提供 的 一 种 解决 方案 称 为 潜在 类 型 机 制 或 结构 化 类 型 机 制 ， 而 更 古怪 的 术语 
称 为 鸭子 类 型 机 制 ， 即 “如 果 它 走 起 来 像 鸭子， 并 且 叫 起 来 也 像 鸭子， 那么 你 就 可 以 将 它 当 作 
了 鸭子 对 待 。” 鸭子 类 型 机 制 变 成 了 一 种 相当 流行 的 术语 ， 可 能 是 因为 它 不 像 其 他 的 术语 那样 承载 
着 历史 的 包 被 。 

泛 型 代码 典型 地 将 在 泛 型 类 型 上 调用 少量 方法 ， 而 具有 潜在 类 型 机 制 的 语言 只 要 求实 现 某 
个 方法 子 集 ， 而 不 是 某 个 特定 类 或 接口 ， 从 而 放松 了 这 种 限制 (并且 可 以 产生 更 加 泛 化 的 代码 ) 。 
正 由 于 此 ， 潜 在 类 型 机 制 使 得 你 可 以 横 跨 类 继承 结构 ， 调 用 不 属于 某 个 公共 接口 的 方法 。 因 此 
实际 上 一 段 代码 可 以 声明 :“ 我 不 关心 你 是 什么 类 型 ， 只 要 你 可 以 speak0 和 sit0 即 可 。.” 由 于 不 
要 求 具体 类 型 ， 因 此 代码 就 可 以 更 加 泛 化 。 

潜在 类 型 机 制 是 一 种 代码 组 织 和 复 用 机 制 。 有 了 它 编写 出 的 代码 相对 于 没有 它 编写 出 的 代 
码 ， 能 够 更 容易 地 复 用 。 代 码 组 织 和 复 用 是 所 有 计算 机 编程 的 基本 手段 : 编写 一 次 ， 多 次 使 用 
并 在 一 个 位 置 保存 代码 。 因 为 我 并 未 被 要 求 去 命名 我 的 代码 要 操作 于 其 上 的 确切 接口 ， 所 以 ， 
有 了 潜在 类 型 机 制 ， 我 就 可 以 编写 更 少 的 代码 ， 并 更 容易 地 将 其 应 用 于 多 个 地 方 。 

两 种 支持 潜在 类 型 机 制 的 语言 实例 是 Python (可 以 从 www.Python.org 免 费 下 载 ) 和 C++e 。 
Python 是 动态 类 型 语言 (事实 上 所 有 的 类 型 检查 都 发 生 在 运行 时 ) ， 而 C++ 是 静态 类 型 语言 (类 
型 检查 发 生 在 编译 期 )， 因 此 潜在 类 型 机 制 不 要 求 静态 或 动态 类 型 检查 。 

如 果 我 们 将 上 面 的 描述 用 Python 来 表示 ， 如 下 所 示 : 

#: generics/DogsAndRobots.py 

class Dog: 

def speak(self): 
print "Arf!" 

def sit(self): 
print "Sitting" 


def reproduce(self): 
pass 


class Robot: 
def speak(self): 
print "Click!" 
def sit(self): 
print "Clank!" 
def oi1Change(self): 
pass 
def perform(anything): 
anything. speak() 
anything. sit() 


a = Dog() 
b = Robot() 
perform(a) 
perform(b) 
fe 


© Ruby 和 Smalltalk 语 言 也 支持 潜在 类 型 机 制 。 
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了 Python 使 用 缩 进来 确定 作用 域 (因此 不 需要 任何 花 括号 ) ， 而 冒号 将 表示 新 的 作用 域 的 开始 。 
P 表示 注释 到 行 尾 ， 就 像 Java 中 的 “/"。 类 的 方法 需要 显 式 地 指定 this 引 用 的 等 价 物 作为 第 一 
个 参数 ， 按 惯例 成 为 self。 构 造 器 调用 不 要 求 任何 类 型 的 “new” 关 键 字 ， 并 且 Python 允许 正则 
( 非 成 员 ) 函数 ， 就 像 perform0 所 表明 的 那样 。 
注意 ， 在 perform(anything) 中 ， 没 有 任何 针对 anything 的 类 型 ，anything 只 是 一 个 标识 符 ， 
它 必须 能 够 执行 perform0 期 望 它 执行 的 操作 ， 因 此 这 里 隐 含 着 一 个 接口 。 但 是 你 从 来 都 不 必 显 
式 地 写 出 这 个 接口 一 一 它 是 潜在 的 。perform0 不 关心 其 参数 的 类 型 ， 因 此 我 可 以 向 它 传递 任何 
对 象 ， 只 要 该 对 象 支持 speak0 和 sit0 方 法 。 如 果 传递 给 perform0 的 对 象 不 支持 这 些 操作 ， 那 么 
将 会 得 到 运行 时 异常 。 
我 们 可 以 用 C++ 产生 相同 的 效果 : 
//: generics/DogsAndRobots.cpp 
class Dog { 
public: 
void speak() {} 
void sit() {} 


void reproduce() {} 
X 


class Robot { 
public: 
void speak() {} 
void sit() {} 
void oilChange() { 
X 
template<class T> void perform(T anything) { 
anything. speak (); 
anything.sit(); 
} 


int main() { 
Dog d: 
Robot r; 
perform(d) ; 
perform(r) ; 
) Mi~ 


在 Python 和 C++ 中 ，Dog 和 Robot 没 有 任何 共同 的 东西 ， 只 是 碰巧 有 两 个 方法 具有 相同 的 签 
名 。 从 类 型 的 观点 看 ， 它 们 是 完全 不 同 的 类 型 。 但 是 ，perform0 不 关心 其 参数 的 具体 类 型 ， 并 
且 潜在 类 型 机 制 允 许 它 接受 这 两 种 类 型 的 对 象 。 

C++ 确保 了 它 实际 上 可 以 发 送 的 那些 消息 ， 如 果 试 图 传递 错误 类 型 ， 编 译 器 就 会 给 你 一 个 错 
误 消 息 ( 这 些 错误 消息 从 历史 上 看 是 相当 可 怕 和 人 宛 长 的 , 而 主要 原因 是 因为 C++ 的 模版 名 声 欠 佳 ) 。 
尽管 它们 是 在 不 同时 期 实现 这 一 点 的 ，C++ 在 编译 期 ， 而 Python 在 运行 时 ， 但 是 这 两 种 语言 都 可 
以 确保 类 型 不 会 被 误 用 ， 因 此 被 认为 是 强 类 型 的 8 。 潜 在 类 型 机 制 没有 损害 强 类 型 机 制 。 

因为 泛 型 是 在 这 场 竞 赛 的 后 期 才 添加 到 Java 中 的 ， 因 此 没有 任何 机 会 可 以 去 实现 任何 类 型 
的 潜在 类 型 机 制 ， 因 此 Java 没 有 对 这 种 特性 的 支持 。 所 以 ， 初 看 起 来 ，Java 的 泛 型 机 制 比 支持 潜 
在 类 型 机 制 的 语言 更 “缺乏 泛 化 性 ”。。 例如 ， 如 果 我 们 试图 用 Java 实 现 上 面 的 示例 ， 那 么 就 会 


被 强制 要 求 使 用 一 个 类 或 接口 ， 并 在 边界 表达 式 中 指定 它 : 


O 因为 你 可 以 使 用 转型 ， 而 转型 实际 上 会 使 类 型 系统 玫 失 能 力 ， 因 此 有 些 人 认为 C++ 是 弱 类 型 的 ， 但 是 这 是 一 
种 极端 情况 。 我 们 可 以 比较 安全 地 说 C++ 是 “具有 通气 门 的 强 类 型 "。 
© Java 使 用 擦 除 的 泛 型 实现 有 时 被 称 为 “第 二 类 泛 型 类 型 "。 
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11: generics/Performs. java 


public interface Performs { 
void speak(); 
void sit(); 

} Mi~ 


41: generics/DogsAndRobots . java 
// No latent typing in Java 

import typeinfo.pets.*; 

import static net.mindview.util.Print.* 





class PerformingDog extends Dog implements Performs { 
public void speak() { print(*Woof! } 
public void sit() { print("Sitting"); 》 
public void reproduce() {} 

} 


class Robot implements Performs { 
public void speak() { print("Cli 
public void sit() { print("Clank!") 
public void oflChange() {} 











} 


class Communicate { 
public static <T extends Performs> 
void perform(T performer) { 
performer. speak() ; 
performer.sit(); 
F 
} 


public class DogsAndRobots { 
public static void main(String[] args) { 
PerformingDog d = new PerformingDog() ; 
Robot r = new Robot(); 
Communicate. perform(d) ; 
Communicate. perform(r); 
} 
} /* Output: 
Woof! 
Sitting 
Click! 
Clank! 
Wh 


但 是 要 注意 ，perform0 不 需要 使 用 泛 型 来 工作 ， 它 可 以 被 简单 地 指定 为 接受 一 个 Performs 
WR: 


//: generics/SimpleDogsAndRobots. java 
// Removing the generic; code still works. 


class CommunicateSimply { 
static void perform(Performs performer) { 
performer. speak () ; 
performer.sit(); 
} 
} 


| public class SimpleDogsAndRobots { 

public static void main(String[] args) { 
CommunicateSimply.perform(new Per formingDog()) ; 
CommunicateSimply.perform(new Robot ()): 


} 
| } /* Output: 
Woof! 
Sitting 
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Click!” 
Clank! 
i~ 


在 本 例 中 ， 泛 型 不 是 必需 的 ， 因 为 这 些 类 已 经 被 强制 要 求实 现 Performs 接 口 。 
15.17 对 缺乏 潜在 类 型 机 制 的 补偿 


尽管 Java 不 支持 潜在 类 型 机 制 ， 但 是 这 并 不 意味 着 有 界 泛 型 代码 不 能 在 不 同 的 类 型 层次 结 
构 之 间 应 用 。 也 就 是 说 ， 我 们 仍旧 可 以 创建 真正 的 泛 型 代码 ， 但 是 这 需要 付出 一 些 额外 的 努力 。 


15.17.1 反射 
可 以 使 用 的 一 种 方式 是 反射 ， 下 面 的 perform0 方 法 就 是 用 了 潜在 类 型 机 制 : 


//: generics/LatentReflection. java 
// Using Reflection to produce latent typing. 
import java.lang.reflect.*; 
import static net.mindview.util.Print.*; 
// Does not implement Performs: 
class Mime { 
public void walkAgainstThewind() {} 
public void sit() { print("Pretending to sit"): } 
Public void pushInvisiblewalls() {} 
Public String toString() { return "Mime"; } 
) 


// Does not implement Performs: 

class SmartDog { 
public void speak() { print("Woof!"); } 
public void sit() { print("Sitting"); } 
public void reproduce() {} 

} 


class ConmunicateReflectively { 
public static void perform(Object speaker) { 
Class<?> spkr = speaker. getClass(): 
try ( 
try { 
Method speak = spkr.getMethod("speak"); 
speak. invoke (speaker) ; 
} catch(NoSuchMethodException e) { 
print(speaker + " cannot speak"); 


} 

try { 
Method sit = spkr.getMethod("sit"); 
sit. invoke (speaker) ; 

} catch(NoSuchMethodException e) { 
print(speaker + “ cannot sit"); 


} 
} catch(Exception e) { 
throw new RuntimeException(speaker.toString(), e): 
} 
} 
} 


public class LatentReflection { 
public static void main(String[] args) { 
CommunicateReflectively.perform(new SmartDog()); 
CommunicateRef lect ively.perform(new Robot ()); 
CommunicateReflectively.perform(new Mime()) ; 


} 
} /* Output: 
Woof! 
Sitting 
Click! 
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Clank! 
Mime cannot speak 
Pretending to sit 
Win 


上 例 中 ， 这 些 类 完全 是 彼此 分 离 的 ， 没 有 任何 公共 基 类 (BRT Object) 或 接口 。 通 过 反射 ， 
CommunicateReflectively.perform0 能 够 动态 地 确定 所 需要 的 方法 是 否 可 用 并 调用 它们 。 它 甚至 
能 够 处 理 Mime 只 具有 一 个 必需 的 方法 这 一 事实 ， 并 能 够 部 分 实现 其 目标 。 

15.17.2 将 一 个 方法 应 用 于 序列 

反射 提供 了 一 些 有 趣 的 可 能 性 ， 但 是 它 将 所 有 的 类 型 检查 都 转移 到 了 运行 时 ， 因 此 在 许多 
情况 下 并 不 是 我 们 所 希望 的 。 如 果 能 够 实现 编译 期 类 型 检查 ， 这 通常 会 更 符合 要 求 。 但 是 有 可 
能 实现 编译 期 类 型 检查 和 潜在 类 型 机 制 吗 ? 

让 我 们 看 一 个 说 明 这 个 问题 的 示例 。 假 设想 要 创建 一 个 apply0 方 法 ， 它 能 够 将 任何 方法 应 
用 于 某 个 序列 中 的 所 有 对 象 。 这 是 接口 看 起 来 并 不 适合 的 情况 ， 因 为 你 想 要 将 任何 方法 应 用 于 
一 个 对 象 集合 ， 而 接口 对 于 描述 “任何 方法 ”存在 过 多 的 限制 。 如 何 用 Java 来 实现 这 个 需求 呢 ? 

最 初 ， 我 们 可 以 用 反射 来 解决 这 个 问题 ， 由 于 有 了 Java SE5 的 可 变 参数 ， 这 种 方式 被 证 明 是 
相当 优雅 的 ， 


//: generics/Apply.java 

// (main: ApplyTest} 

import java. lang.reflect.*; 

import java.util.*: 

import static net.mindview.util.Print.*; 


public class Apply { 
public static <T, S extends Iterable<? extends T>> 
void apply(S seq, Method f, Object... args) { 
try { 
for(T t: seq) 
f.invoke(t, args); 
) catch(Exception e) { . 
// Failures are programmer errors 
throw new RuntimeExcept ion(e) ; 
} 
} 
} 


class Shape { 
public void rotate() { print(this + " rotate"); } 
public void resize(int newSize) { 
print(this + " resize ”+ newSize); 


7 
class Square extends Shape {} 


class FilledList<T> extends ArrayList<T> { 
public FilledList(Class<? extends T> type, int size) { 
try { 
for(int i = @; i < size; i++) 
// Assumes default constructor: 
add(type.newInstance()); 
} catch(Exception e) { 
throw new RuntimeException(e); 
} 
} 
} 


class ApplyTest { 
public static void main(String{] args) throws Exception { 
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List<Shape> shapes = new ArrayList<Shape>(); 
for(int i = @; 1 < 10; i++) 

shapes.add(new Shape()); 

Apply. apply(shapes, Shape.class.getethod(*rotate")) ; 
Apply. apply (shapes, 

Shape.class.getMethod(“resize”, int.class), 5); 
List<Square> squares = new ArrayList<Square>(); 
for(int i = 0; i < 10; i++) 

squares. add(new Square()): 

Apply .apply(squares, Shape.class.getMethod("rotate")); 
Apply. apply (squares, 
Shape.class.getMethod("resize", int.class), 5); 


Apply .apply(new FilledList<Shape>(Shape.class, 10), 
Shape.class.getMethod("rotate")); 

Apply .apply(new FilledList<Shape>(Square.class, 16), 
Shape.class.getMethod("rotate")); 


SimpleQueue<Shape> shapeQ = new SimpleQueue<Shape>(); 
for(int 1 = 0; i < 5; i++) { 

shapeQ.add(new Shape()); 

shapeQ.add(new Square()); 


} 
Apply.apply(shapeQ, Shape.class.getMethod("rotate")); 


} /* (Execute to see output) *///:~ 


在 Apply 中 ， 我 们 运气 很 好 ， 因 为 碰巧 在 Java 中 内 建 了 一 个 由 Java 容 器 类 库 使 用 的 Iterable 接 
只 。 正 由 于 此 ，apply0 方 法 可 以 接受 任何 实现 了 Iterable 接 口 的 事物 ， 包 括 诸如 List 这 样 的 所 有 
Collection 类 。 但 是 它 还 可 以 接受 其 他 任何 事物 ， 只 要 能 够 使 这 些 事物 是 Iterable 的 一 例如， 在 
main0 中 使 用 的 下 面 定义 的 SimpleQueue 类 : 


//: generics/SimpleQueue. java 
// A different kind of container that is Iterable 
import java.util.*; 


public class SimpleQueue<T> implements Iterable<T> { 
Private LinkedList<T> storage = new LinkedList<T>(); 
Public void add(T t) { storage.offer(t); } 
public T get() { return storage.poll(); } 
public Iterator<T> iterator() { 
return storage. iterator (); 


} 

} Mi~ 

在 Applyjava 中 ， 异 常 被 转换 为 RuntimeException， 因 为 没有 多 少 办 法 可 以 从 这 种 异常 中 恢 
复 一 在 这 种 情况 下 ， 它 们 实际 上 代表 着 程序 员 的 错误 。 

注意 ， 我 必须 放置 边界 和 通配符 ， 以 便 使 Apply 和 FilledList 在 所 有 需要 的 情况 下 都 可 以 使 用 。 
可 以 试验 一 下 ， 将 这 些 边界 和 通配符 拿 出 来 ， 你 就 会 发 现 某 些 Apply 和 FilledList 应 用 将 无 法 工作 。 

FilledList 表 示 有 点 进退 两 难 的 情况 。 为 了 使 某 种 类 型 可 用 ， 它 必须 有 默认 (FEB) 构造 器 ， 
但 是 Java 没 有 任何 方式 可 以 在 编译 期 断言 这 种 事情 ， 因 此 这 就 变 成 了 一 个 运行 时 问题 。 确 保 编 
译 期 检查 的 常见 建议 是 定义 一 个 工厂 接口 ， 它 有 一 个 可 以 生成 对 象 的 方法 ， 然 后 FilledList 将 接 
受 这 个 接口 而 不 是 这 个 类 型 标记 的 “原生 工厂 "， 而 这 样 做 的 问题 是 在 FilledList 中 使 用 的 所 有 类 
都 必须 实现 这 个 工厂 接口 。 唉 ， 大 多 数 的 类 都 是 在 不 了 解 你 的 接口 的 情况 下 创建 的 ， 因 此 也 就 
没有 实现 这 个 接口 。 稍 后 ， 我 将 展示 一 种 使 用 适配器 的 解决 方案 。 

但 是 上 面 所 展示 的 使 用 类 型 标记 的 方法 可 能 是 一 种 合理 的 折 中 (至 少 是 一 种 马上 就 能 想到 
解决 方案 )。 通 过 这 种 方式 ， 使 用 像 FilledList 这 样 的 东西 就 会 非常 容易 ， 我 们 会 马上 想到 要 使 用 
它 而 不 是 会 忽略 它 。 当 然 ， 因 为 错误 是 在 运行 时 报告 的 ， 所 以 你 要 有 把 握 ， 这 些 错误 将 在 开发 
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过 程 的 早期 出 现 。 

注意 ， 类 型 标记 技术 是 Java 文 献 推荐 的 技术 ， 例 如 Gilad Bracha 在 他 的 论文 《Generics in Java 
Programming Language) ° 中 写 道 “这 是 一 种 惯用 法 ， 例 如 ， 在 操作 注解 的 新 API 中 得 到 了 广泛 
的 应 用 "。 但 是 ， 我 发 现 人 们 对 这 种 技术 的 适应 程度 不 一 ， 有 些 人 强烈 地 首选 本 章 前 面 描 述 的 工 
厂 方式 。 

尽管 Java 解决 方案 被 证 明 很 优雅 ， 但 是 我 们 必须 知道 使 用 反射 (尽管 反射 在 最 近 版 本 的 Java 
中 已 经 明显 地 改善 了 ) 可 能 比 非 反射 的 实现 要 慢 一 些 ， 因 为 有 太 多 的 动作 都 是 在 运行 时 发 生 的 。 
这 不 应 该 阻止 你 使 用 这 种 解决 方案 的 脚步 ， 至 少 可 以 将 其 作为 一 种 马上 就 能 想到 的 解决 方案 
(以 防止 陷入 不 成 熟 的 优化 中 ) ， 但 这 毫 无 疑问 是 这 两 种 方法 之 间 的 一 个 差异 。 

练习 40: (3) 向 typeinfojava 中 的 所 有 宠物 中 添加 一 个 speak() 方 法 。 修 改 Applyjava， 使 得 
我 们 可 以 对 Pet 的 异 构 集合 调用 speak0 。 
15.17.3 当 你 并 未 碰巧 拥有 正确 的 接口 时 

上 面 的 示例 是 受益 的 ， 因 为 Iterable 接 口 已 经 是 内 建 的 ， 而 它 正 是 我 们 需要 的 。 但 是 更 一 般 
的 情况 又 会 怎样 呢 ? 如 果 不 存 在 刚好 适合 你 的 需求 的 接口 呢 ? 

例如 ， 让 我 们 泛 化 FilledList 中 的 思想 ， 创 建 一 个 参数 化 的 方法 fil0)， 它 接受 一 个 序列 ， 并 
使 用 Generator 填 充 它 。 当 我 们 尝试 着 用 Java 来 编写 时 ， 就 会 陷入 问题 之 中 ， 因 为 没有 任何 像 前 
面 示例 中 的 iterable 接 口 那样 的 “Addable” 便 利 接口 。 因 此 你 不 能 说 :“ 可 以 在 任何 事物 上 调用 
add0。 而 必须 说 :“ 可 以 在 Collection 的 子 类 型 上 调用 add0.。” 这 样 产生 的 代码 并 不 是 特别 省 化 ， 
因为 它 必须 限制 为 只 能 工作 于 Collection 实 现 。 如 果 我 试图 使 用 没有 实现 Collection 的 类 ， 那 么 我 
的 泛 化 代码 将 不 能 工作 。 下 面 是 这 段 代码 的 样子 : 


//: generics/Fill.java 

// Generalizing the FilledList idea 
1 (main: FillTest} 

import java.util.*; 


// Doesn't work with “anything that has an add()." There is 
// no “Addable" interface so we are narrowed to using a 

// Collection. We cannot generalize using generics in 

// this case. 


public class Fill { 
public static <T> void fill(Collection<T> collection, 
Class<? extends T> classToken, int size) { 
for(int i = 0; i < size; i++) 
// Assumes default constructor: 
try { 
collection. add(classToken.newInstance()): 
} catch(Exception e) { 
throw new RuntimeException(e); 
} 
} 


class Contract { 
private static long counter = @; 
private final long id = counter++; 
public String toString() { 
return getClass().getName() + " + id; 


} 
class TitleTransfer extends Contract {} 


日 参见 本 章 未 尾 的 引用 。 
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class FillTest { 
public static void main(String[] args) { 
List<Contract> contracts = new ArrayList<Contract>(); 
Fill.fill(contracts, Contract.class, 3): 
Fill.fill(contracts, TitleTransfer.class, 2); 
for (Contract c: contracts) 
System.out.printin(c) ; 
SimpleQueue<Contract> contractQueue = 
new SimpleQueue<Contract>(); 
// Won't work. fill() is not generic enough: 
// Fill. fill(contractQueue, Contract.class, 3); 
$ 
} /* Output: 
Contract @ 
Contract 1 
Contract 2 
TitleTransfer 3 
TitleTransfer 4 
Wh 


这 正 是 具有 潜在 类 型 机 制 的 参数 化 类 型 机 制 的 价值 所 在 ， 因 为 你 不 会 受 任何 特定 类 库 的 创 
建 者 过 去 所 作 的 设计 决策 的 支配 ， 因 此 不 需要 在 每 次 碰 到 一 个 没有 考虑 到 你 的 具体 情况 的 新 类 
库 时 ， 都 去 重 写 代码 (因此 这 样 的 代码 才 是 真正 “ 泛 化 的 ")。 在 上 面 的 情况 中 ， 因 为 Java 设 计 
者 (可 以 理解 地 ) 没有 预见 到 对 “Addable” 接 口 的 需要 ， 所 以 我 们 被 限制 在 Collection 继 承 层次 
结构 之 内 ， 即 便 SimpleQueue 有 一 个 add0 方 法 ， 它 也 不 能 工作 。 因 为 这 会 将 代码 限制 为 只 能 工 
作 于 Collection， 因 此 这 样 的 代码 不 是 特别 “ 泛 化 "。 有 了 潜在 类 型 机 制 ， 情 况 就 会 不 同 了 。 


15.17.4 用 适配器 仿真 潜在 类 型 机 制 

Java 泛 型 并 不 没有 潜在 类 型 机 制 ， 而 我 们 需要 像 潜在 类 型 机 制 这 样 的 东西 去 编写 能 够 跨 类 
边界 应 用 的 代码 (也 就 是 “ 泛 化 ”代码 )。 存 在 某 种 方式 可 以 绕 过 这 项 限制 吗 ? 

潜在 类 型 机 制 将 在 这 里 实现 什么 ? 它 意味 着 你 可 以 编写 代码 声明 :“ 我 不 关心 我 在 这 里 使 用 
的 类 型 ， 只 要 它 具有 这 些 方法 即 可 。” 实 际 上 ， 潜 在 类 型 机 制 创建 了 一 个 包含 所 需 方法 的 隐 式 接 
口 。 因 此 它 遵 循 这 样 的 规则 ， 如 果 我 们 手工 编写 了 必需 的 接口 (因为 Java 并 没有 为 我 们 做 这 些 
事 )， 那 么 它 就 应 该 能 够 解决 问题 。 

从 我 们 拥有 的 接口 中 编写 代码 来 产生 我 们 需要 的 接口 ， 这 是 适配器 设计 模式 的 一 个 典型 示 
例 。 我 们 可 以 使 用 适配器 来 适 配 已 有 的 接口 ， 以 产生 想 要 的 接口 。 下 面 这 种 使 用 前 面 定义 的 
Coffee 继 承 结构 的 解决 方案 演示 了 编写 适配器 的 不 同方 式 : 


//: generics/Fill2. java 
// Using adapters to simulate latent typing. 
17 (main: Fill2Test} 

import generics.coffee.*; 

import java.util.*; 

import net.mindview.util.*; 

import static net.mindview.util.Print.*; 


interface Addable<T> { void add(T t); } 


public class Fill2 { 
// Classtoken version: 
public static <T> void fill(Addable<T> addable, 
Class<? extends T> classToken, int size) { 
for(int i = @; 1 < size: i++) 
try { 
addable. add(classToken.newInstance()); 
} catch(Exception e) { 
throw new RuntimeException(e); 
j 
$ 
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// Generator version: 
public static <T> void fill(Addable<T> addable, 
Generator<T> generator, int size) { 
for(int i = 8; i < size; i++) 
addable. add (generator .next()): 
} 


} 


// To adapt a base type, you must use composition. 
// Make any Collection Addable using composition: 
class AddableCollectionAdapter<T> implements Addable<T> { 
private Collection<T> c; 
public AddableCollectionAdapter (Collection<T> c) { 
this.c = c; 
} 
public void add(T item) { c.add(item); } 
} 
// A Helper to capture the type automatically: 
class Adapter { 
public static <T> 
Addable<T> collect ionAdapter (Collection<T> c) { 
return new AddableCotlectionAdapter<T>(c) ; 
} 


// To adapt a specific type, you can use inheritance. 

// Make a SimpleQueue Addable using inheritance: 

class AddableSimpleQueue<T> 

extends SimpleQueue<T> implements Addable<T> { 
public void add(T item) { super.add(item); } 

} 


class Fill2Test { 
public static void main(String(} args) { 
// Adapt a Collection: 
List<Coffee> carrier = new ArrayList<Coffee>(); 
FNR. HU 
new AddableCol Lect ionAdapter<Cof fee> (carrier), 
Coffee.class, 3); 
// Helper method captures the type: 
Fill2.fill (Adapter .col Lect fonAdapter (carrier), 
Latte.class, 2); 
for (Coffee c: carrier) 
print(c); 
prfnt("- 
// Use an adapted class: 
AddableSimpleQueue<Coffee> cof feequeue = 
new Addables impleQueue<Coffee>(); 
Fi112. fill (coffeeQueue, Hocha.class, 4); 
Fi112. Fill (coffeeQueue, Latte.class, 1); 
for (Coffee c: coffeeQueue) 
print(c); 
} 


} /* Output: 
Coffee 6 
Coffee 1 
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Fil2 对 Collection 的 要 求 与 F 训 不 同 ， 它 只 需要 实现 了 Addable 的 对 象 ， 而 Addable 已 经 为 Fi 
编写 了 一 一 它 是 我 希望 编译 器 帮 我 创建 的 潜在 类 型 的 一 种 体现 。 

在 这 个 版 本 中 ， 我 还 添加 了 一 个 重 载 的 fill()， 它 接受 一 个 Generator 而 不 是 类 型 标记 。 
Generator 在 编译 期 是 类 型 安全 的 : 编译 器 将 确保 传递 的 是 正确 的 Generator， 因 此 不 会 抛 出 任 
何 异常 。 

第 一 个 适配器 ，AddableCollectionAdapter， 可 以 工作 于 基 类 型 Collection， 这 意味 着 
Collection 的 任何 实现 都 可 以 使 用 。 这 个 版 本 直接 存储 Collection 引 用 ， 并 使 用 它 来 实现 add0。 

如 果 有 一 个 具体 类 型 而 不 是 继承 结构 的 基 类 ， 那 么 当 使 用 继承 来 创建 适配器 时 ， 你 可 以 稍 
微 少 编写 一 些 代码 ， 就 像 在 AddableSimpleQueue 中 看 到 的 那样 。 

在 Fill2Test.main0 中 ， 你 可 以 看 到 各 种 不 同类 型 的 适配器 在 运行 。 首 先 ，Colleetion 类 型 是 
由 AddableCollectionAdapter 适 配 的 。 这 个 第 二 个 版 本 使 用 了 一 个 泛 化 的 辅助 方法 ， 你 可 以 看 到 
这 个 泛 化 方法 是 如 何 捕获 类 型 并 因此 而 不 必 显 式 地 写 出 来 的 一 -这 是 产生 更 优雅 的 代码 的 一 种 
惯用 技巧 。 

接 下 来 ， 使 用 了 预 适 配 的 AddableSimpleQueue。 注 意 ， 在 两 种 情况 下 ， 适 配器 都 允许 前 面 
没有 实现 Addable 的 类 用 于 Fill2.fill0 中 。 

使 用 像 这 样 的 适配器 看 起 来 是 对 缺乏 潜在 类 型 机 制 的 一 种 补偿 ， 因 此 允许 编写 出 真正 的 泛 
化 代码 。 但 是 ， 这 是 一 个 额外 的 步骤 ， 并 且 是 类 库 的 创建 者 和 消费 者 都 必须 理解 的 事物 ， 而 缺 
乏 经 验 的 程序 员 可 能 还 没有 能 够 掌握 这 个 概念 。 潜 在 类 型 机 制 通过 移 除 这 个 额外 的 步 又， 使 得 

736) 泛 化 代码 更 容易 应 用 ， 这 就 是 它 的 价值 所 在 。 
练习 41: (1) 修改 Fil2java， 用 typeinfojava 中 的 类 取代 Coffee 中 的 类 。 


15.18 将 函数 对 象 用 作 策 略 


最 后 一 个 示例 通过 使 用 前 面 一 节 描述 的 适配器 方式 创建 了 真正 泛 化 的 代码 。 这 个 示例 开始 
时 是 一 种 尝试 ， 要 创建 一 个 元 素 序 列 的 总 和 ， 这 些 元 素 可 以 是 任何 可 以 计算 总 和 的 类 型 ， 但 是 ， 
后 来 这 个 示例 使 用 功能 型 编程 风格 ， 演 化 成 了 可 以 执行 通用 的 操作 。 

如 果 只 查看 尝试 添加 对 象 的 过 程 ， 就 会 看 到 这 是 在 多 个 类 中 的 公共 操作 ， 但 是 这 个 操作 没 
有 在 任何 我 们 可 以 指定 的 基 类 中 表示 一 一 有 时 甚至 可 以 使 用 “+” 操 作 府 ， 而 其 他 时 间 可 以 使 用 
某 种 add 方 法 。 这 是 在 试图 编写 泛 化 代码 的 时 候 通 常会 碰 到 的 情况 ， 因 为 你 想 将 这 些 代码 应 用 于 
多 个 类 上 一 特别 是 ， 像 本 例 一 样 ， 作 用 于 多 个 已 经 存在 且 我 们 不 能 “修正 ”的 类 上 。 即 使 你 
可 以 将 这 种 情况 窗 化 到 Number 的 子 类 ， 这 个 超 类 也 不 包括 任何 有 关 “ 可 添加 性 ”的 东西 。 

解决 方案 是 使 用 策略 设计 模式 ， 这 种 设计 模式 可 以 产生 更 优雅 的 代码 ， 它 将 “变化 的 
事物 ”完全 隔离 到 了 一 个 函数 对 象 中 ” 。 函 数 对 象 就 是 在 某 种 程度 上 行为 像 函 数 的 对 象 一 一般 
地 ， 会 有 一 个 相关 的 方法 (在 支持 操作 符 重 载 的 语言 中 ， 可 以 创建 对 这 个 方法 的 调用 ， 而 这 个 
调用 看 起 来 就 和 普通 的 方法 调用 一 样 )。 函 数 对 象 的 价值 就 在 于 ， 与 普通 方法 不 同 ， 它 们 可 以 传 
递 出 去 ， 并 且 还 可 以 拥有 在 多 个 调用 之 间 持 久 化 的 状态 。 当 然 ， 可 以 用 类 中 的 任何 方法 来 实现 
与 此 相似 的 操作 ， 但 是 (与 使 用 任何 设计 模式 一 样 ) 函数 对 象 主要 是 由 其 目的 来 区 别 的 。 这 里 
的 目的 就 是 要 创建 某 种 事物 ， 使 它 的 行为 就 像 是 一 个 可 以 传递 出 去 的 单个 方法 一 样 ， 这 样 ， 它 

E ”就 和 策略 设计 模式 紧 而 合 了 ， 有 时 甚至 无 法 区 分 。 














O 有 时 会 看 到 将 它们 称 为 “ 仿 函 数 "， 我 将 使 用 术语 “函数 对 象 ”而 不 是 “ 仿 函 数 "， 因 为 术语 “ 仿 函数 ”在 数 
学 中 有 具体 而 不 同 的 意义 。 
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尽管 可 以 发 现 我 使 用 了 大 量 的 设计 模式 ， 但 是 在 这 里 它们 之 间 的 界限 是 模糊 的 : 我 们 在 创 
建 执行 适 配 操作 的 函数 对 象 ， 而 它们 将 被 传递 到 用 作 策 略 的 方法 中 。 

通过 采用 这 种 方式 ， 我 添加 了 最 初 着 手 创建 的 各 种 类 型 的 泛 型 方法 ， 以 及 其 他 的 泛 型 方法 。 
下 面 是 产生 的 结果 : 


//: generics/Functional.java 
import java.math.*; 

import java.util.concurrent.atomic. 
import java.util.*; 
import static net.mindview.util.Print.*; 








// Different types of function objects: 
interface Combiner<T> { T combine(T x, T y); } 
interface UnaryFunction<R,1> { R function(T x); } 
interface Collector<T> extends UnaryFunction<T.T> { 

T result(); // Extract result of collecting parameter 


t 
‘interface UnaryPredicate<T> { boolean test(T x); } 


public class Functional { 
// Calls the Combiner object on each element to combine 
// it with a running result, which is finally returned: 
public static <T> T 
reduce(Iterable<T> seq, Combiner<T> combiner) { 
Iterator<T> it = seq. iterator(); 
if(it.nasNext()) { 
T result = it.next(); 
while(it.hasNext()) 
result = combiner.combine(result, it.next()): 
return result; 
} 
// Tf seq is the empty list: 
return null; // Or throw exception 
} 
// Take a function object and call it on each object in 
// the list, ignoring the return value. The function 
// object may act as a collecting parameter, so it is 
// returned at the end. 
public static <T> Collector<T> 
forEach(Iterable<T> seq, Collector<T> func) { 
for(T t : seq) 
func. function(t); 738) 
return func; 
} 
// Creates a list of results by calling a 
// function object for each object in the list: 
public static <R,T> List<R> 
transform(Iterable<T> seq, UnaryFunction<R,T> func) { 
List<R> result = new ArrayList<R>(); 
for(T t : seq) 
result.add(func. function(t)) ; 
return result; 
} 
// Applies a unary predicate to each item in a sequence, 
// and returns a list of items that produced "true": 
public static <T> List<T> 
filter(Iterable<T> seq. UnaryPredicate<T> pred) { 
List<T> result = new ArrayList<T>(); 
for(T t : seq) 
if(pred.test(t)) 
result.add(t): 
return result; 
} 
// To use the above generic methods, we need to create 
// function objects to adapt to our particular needs: 
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static class IntegerAdder implements Combiner<Integer> { 
public Integer combine(Integer x, Integer y) { 
return x + y; 
} 
} 
static class 
IntegerSubtracter implements Combiner<Integer> { 
public Integer combine(Integer x, Integer y) { 
return x - y; 
} 
} 
static class 
BigDecimalAdder implements Combiner<BigDecimal> { 
public BigDecimal combine (BigDecimal x, BigDecimal y) { 
return x.add(y); 
} 
} 
static class 
BigIntegerAdder implements Combiner<BigInteger> { 
public BigInteger combine(BigInteger x, BigInteger y) { 
return x.add(y) ; 


} 
} 
static class 
AtomicLongAdder implements Combiner<AtomicLong> { 
public AtomicLong combine(AtomicLong x, AtomicLong y) { 
// Not clear whether this is meaningful: 
return new AtomicLong(x.addAndGet (y. get ())): 
} 
} 
// We can even make a UnaryFunction with an "utp" 
/1/ (Units in the last place): 
static class BigDecimalU1p 
implements UnaryFunction<Bigdecimal ,BigDecimal> { 
public BigDecimal function(Bigdecimal x) ( 
return x.ulp(); 
} 


} 
static class GreaterThan<T extends Comparable<T>> 
implements UnaryPredicate<T> { 
private T bound; 
public GreaterThan(T bound) { this.bound = bound; } 
public boolean test(T x) { 
return x.compareTo(bound) > @; 
} 
} 
static class MultiplyingIntegerCollector 
implements Collector<Integer> { 
private Integer val = 1; 
public Integer function(Integer x) { 
val *= x; 
return val; 
} 
public Integer result() { return val; } 
} 
public static void main(String(] args) { 
// Generics, varargs & boxing working together: 
List<Integer> li = Arrays.asList(1, 2, 3. 4. 5, 6, 7); 
Integer result = reduce(li, new IntegerAdder()); 
print(result); 


result = reduce(1i, new IntegerSubtracter()); 
print(result); 


print(filter(li, new GreaterThan<Integer>(4))); 


print (forEach(1i, 
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new MultiplyingIntegerCollector()).result()); 


print(forEach(filter(1li, new GreaterThan<Integer>(4)), 
new MultiplyingIntegerCollector()).result()); 


MathContext mc = new MathContext(7); 
List<BigDecimal> lbd = Arrays.asList( 
new BigDecimal (1.1, mc), new BigDecimal (2.2, mc), 
new BigDecimal (3.3, mc), new BigDecimal(4.4, mc)); 
BigDecimal rbd = reduce(1bd, new BigDecimalAdder()); 
print (rbd) ; 








print(filter (lbd, 
new GreaterThan<BigDecimal> (new BigDecimal(3)))); 


// Use the prime-generation facility of BigInteger: 
List<BigInteger> lbi = new ArrayList<BigInteger>(); 
BigInteger bi = BigInteger. valueOf (11); 
for(int i = 0; i < 11; i++) { 

lbi add (bi); 

bi = bi.nextProbablePrime(); 


} 
print (tbi); 


BigInteger rbi = reduce(lbi, new BigIntegerAdder()); 
print(rbi); 

// The sum of this list of primes is also prime: 
print(rbi.isProbablePrime(5)); 


List<AtomicLong> lal = Arrays.asList( 

new AtomicLong(11), new AtomicLong(47), 

new Atomiclong(74), new AtomicLong(133)); 
AtomicLong ral = reduce(lal, new AtomicLongAdder()); 
print(ral); 


print (transform(1bd,new BigDecimalUlp())): 
} 
} /* Output: 
28 o. 


-26 
【5，6，7] 
5040 


210 

11.000000 

13.300000, 4.400000) 

[11, 13, 17, 19, 23, 29, 31, 37, 41; 43. 47) 
311 

true 


265 
[0.0000901, 0.000001, 0.000001, @.006001] 
Ii~ 


我 是 从 为 不 同类 型 的 函数 对 象 定义 接口 开始 的 ， 这 些 接口 都 是 按 需 创建 的 ， 因 为 我 为 每 个 接 
口 都 开发 了 不 同 的 方法 ， 并 发 现 了 每 个 接口 的 需求 。Combiner 类 是 由 一 位 不 知名 的 作者 在 我 的 
Web 网 站 上 贴 出 的 文章 建议 构建 的 。Combiner 抽 象 挤 了 将 两 个 对 象 添加 在 一 起 的 具体 细节 ， 并 且 
只 是 声明 它们 在 某 种 程度 上 被 结合 在 一 起 。 因 此 ， 可 以 看 到 ，IntegerAdder 和 IntegerSubstract 可 
以 是 Combiner 类 型 。 

UnaryFunction 接 受 单一 的 参数 ， 并 产生 一 个 结果 ， 这 个 参数 和 结果 不 需要 是 相同 的 类 型 。 
Collector 被 用 作 “ 收 集 参 数 "， 并 且 当 你 完成 时 ， 可 以 从 中 抽取 结果 。UnaryPredicate 将 产生 一 
个 boolean 类 型 的 结果 。 还 可 以 创建 其 他 类 型 的 函数 对 象 ， 但 是 这 些 已 经 足够 说 明 问 题 了 。 

Functional 类 包含 大 量 的 泛 型 方法 ， 它 们 可 以 将 函数 对 象 应 用 于 序列 。reduce0 将 Combiner 




















[742] 














(743| 











430 BSE 





中 的 函数 应 用 于 序列 中 的 每 个 元 素 ， 以 产生 单一 的 结果 。 

foreach0 接 受 一 个 Collector， 并 将 其 函数 应 用 于 每 个 元 素 ， 但 同时 会 忽略 每 次 函数 调用 的 
结果 。 这 只 能 被 称 为 是 副作用 (这 不 是 “功能 型 ”编程 风格 ,但 仍旧 是 有 用 的 )， 或 者 我 们 可 以 
让 Collector 维 护 内 部 状态 ， 从 而 变 成 一 个 收集 参数 ， 就 像 在 本 例 中 看 到 的 那样 。 

transform0 通 过 在 序列 中 的 每 个 对 象 上 调用 UnaryFunetion， 并 捕获 调用 结果 ， 来 产生 一 个 
列表 。 

最 后 ，fter0 将 UnaryPredicate 应 用 到 序列 中 的 每 个 对 象 上 ， 并 将 那些 返回 true 的 对 象 存储 
到 一 个 List 中 。 

可 以 定义 附加 的 泛 型 函数 ， 例 如 ，C++ STL 就 具有 很 多 这 类 函数 。 在 诸如 JGA (Generic 
Algorithms for Java) 这 样 的 开源 类 库 中 ， 这 个 问题 也 解决 了 。 

在 C++ 中 ,潜在 类 型 机 制 将 在 你 调用 函数 时 负责 协调 各 个 操作 ， 但 是 在 Java 中 ， 我 们 需要 编 
写 函数 对 象 来 将 泛 型 方法 适 配 为 我 们 特定 的 需求 。 因 此 ， 这 个 类 接 下 来 的 部 分 展示 了 函数 对 象 
的 各 种 不 同 的 实现 。 例 如 ， 注 意 ，IntegerAdder 和 BigDecimalAdder 通 过 为 它们 特定 的 类 型 调用 
恰当 的 方法 ， 从 而 解决 了 相同 的 问题 ， 即 添加 两 个 对 象 。 因 此 ， 这 是 适配器 模式 和 策略 模式 的 
结合 。 

在 main0 中 ， 你 可 以 看 到 ， 在 每 个 方法 调用 中 ， 都 会 传递 一 个 序列 和 适当 的 函数 对 象 。 还 
有 大 量 的 、 可 能 会 相当 复杂 的 表达 式 ， 例 如 : 


forEach(fttter(11，new GreaterThan(4)), 
new HuttiptyingIntegerCottector()) .resutt() 


这 将 通过 选取 1i 中 大 于 4 的 所 有 元 素 而 产生 一 个 列表 ， 然 后 将 MultiplyingIntegerCollector0 
应 用 于 所 产生 的 列表 ， 并 抽取 result0。 我 不 会 再 解释 剩余 代码 的 细节 了 ， 通 过 通读 它们 你 就 可 
以 了 解 它们 的 作用 。 

练习 42: (5) 创建 两 个 独立 的 类 ， 它 们 没有 任何 共同 的 东西 。 每 个 类 都 应 该 持 有 一 个 值 ， 并 
至 少 有 产生 这 个 值 和 在 这 个 值 上 执行 修改 的 方法 。 修 改 Functionaljava， 使 它 可 以 在 由 你 的 类 构 
成 的 集合 上 执行 函数 型 操作 (这 些 操作 不 必 像 Functionaljava 中 的 操作 那样 是 算术 型 的 )。 


15.19 总 结 : 转型 真 的 如 此 之 糟 吗 ? 


自从 C++ 模版 出 现 以 来 ， 我 就 一 直 在 致力 于 解释 它 ， 我 可 能 比 大 多 数 人 都 更 早 地 提出 了 下 
面 的 论点 。 直 到 最 近 ， 我 才 停 下 来 ， 去 思考 这 个 论点 到 底 在 多 少时 间 内 是 有 效 的 一 一 我 将 要 描 
述 的 问题 到 底 有 多 少 次 可 以 穿越 障碍 得 以 解决 。 

这 个 论点 就 是 :使 用 泛 型 类 型 机 制 的 最 吸引 人 的 地 方 ， 就 是 在 使 用 容器 类 的 地 方 ， 这 些 类 
包括 诸如 各 种 List、 各 种 Set、 各 种 Map 等 你 在 第 11 章 中 看 到 的 ， 和 你 将 在 第 17 章 中 看 到 的 各 种 
类 。 在 Java SE5 之 前 ， 当 你 将 一 个 对 象 放 置 到 容器 中 时 ， 这 个 对 象 就 会 被 向 上 转型 为 Object， 因 
此 你 会 丢失 类 型 信息 。 当 你 想 要 将 这 个 对 象 从 容器 中 取 回 ， 用 它 去 执行 某 些 操作 时 ， 必 须 将 其 
向 下 转型 回 正确 的 类 型 。 我 用 的 示例 是 持 有 Cat 的 List (这 个 示例 的 一 种 使 用 苹果 和 桔子 的 变 体 
在 第 11 章 的 开头 展示 过 )。 如 果 没 有 Java SE5 的 泛 型 版 本 的 容器 ， 你 放 到 容器 里 的 和 从 容器 中 取 
回 的 ， 都 是 Object。 因 此 ， 我 们 很 可 能 会 将 一 个 Dog 放 置 到 Cat 的 List 中 。 . 

但 是 , 泛 型 出 现 之 前 的 Java 并 不 会 让 你 误 用 放 入 到 容器 中 的 对 象 。 如 果 将 一 个 Dog 扔 到 Cat 
的 容器 中 ， 并 且 试图 将 这 个 容器 中 的 所 有 东西 都 当 作 Cat 处 理 ， 那 么 当 你 从 这 个 Cat 容 器 中 取 回 
那个 Dog 引 用 ， 并 试图 将 其 转型 为 Cat 时 ， 就 会 得 到 一 个 RuntimeException。 你 仍旧 可 以 发 现 问 
题 ,但 是 是 在 运行 时 而 非 编译 期 发 现 它 的 。 
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在 本 书 以 前 的 版 本 中 ， 我 曾经 说 过 : 

这 不 止 是 令 人 恼火 ， 它 还 可 能 会 产生 难以 发 现 的 缺陷 。 如 果 这 个 程序 的 某 个 部 分 (或 数 个 
部 分 ) 向 容器 中 插入 了 对 象 ， 并 且 通 过 异常 ， 你 在 程序 的 另 一 个 独立 的 部 分 中 发 现 有 不 良 对 象 
被 放置 到 了 容器 中 ， 那 么 必须 发 现 这 个 不 良 插入 到 底 是 在 何 处 发 生 的 。 

但 是 ， 随 着 对 这 个 论点 的 进一步 检查 ， 我 开始 怀疑 它 了 。 首 先 ， 这 会 多 么 频繁 地 发 生 呢 ? 
我 记得 这 类 事情 从 未 发 生 在 我 身上 ， 并 且 当 我 在 会 议 上 询问 其 他 人 时 ， 我 也 从 来 没有 听 说 过 有 
人 碰 上 过 。 另 一 本 书 使 用 了 一 个 示例 ， 它 是 一 个 包含 String 对 象 的 被 称 为 iles 的 列表 在 这 个 示例 
中 ， 向 files 中 添加 一 个 File 对 象 看 起 来 相当 自然 ， 因 此 这 个 对 象 的 名 字 可 能 叫 人 eNames 更 好 。 无 
论 Java 提 供 了 多 少 类 型 检查 ， 仍 旧 可 能 会 写 出 降 涩 的 程序 ， 而 编写 差劲 儿 的 程序 即便 可 以 编译 ， 
它 仍 旧 是 编写 差劲 儿 的 程序 。 可 能 大 多 数 人 都 会 使 用 命名 良好 的 容器 ， 例 如 cats， 因 为 它们 可 以 
向 试图 添加 非 Cat 对 象 的 程序 员 提供 可 视 的 警告 。 并 且 即 便 这 类 事情 发 生 了 ， 它 真正 又 能 潜伏 多 
AW? 只 要 你 开始 用 真实 数据 来 运行 测试 ， 就 会 非常 快 地 看 到 异常 。 

有 一 位 作者 其 至 断言 ， 这 样 的 缺陷 将 “潜伏 数 年 "。 但 是 我 不 记得 有 任何 大 量 的 相关 报告 ， 
来 说 明 人 们 在 查找 “ 狗 在 猫 列表 中 ”这 类 缺陷 时 困难 重重 ， 或 者 是 说 明 人 们 会 非常 频繁 地 产生 
这 种 错误 。 然 而 ， 你 将 在 第 21 章 中 看 到 ， 在 使 用 线程 时 ， 出 现 那 些 可 能 看 起 来 极 罕见 的 缺陷 ， 
是 很 寻常 并 容易 发 生 的 事 ， 而 且 ， 对 于 到 底 出 了 什么 错 ， 这 些 缺 陷 只 能 给 你 一 个 很 模糊 的 概念 。 
因此 ， 对 于 谤 型 是 添加 到 Java 中 的 非常 显著 和 相当 复杂 的 特性 这 一 点 ,“ 狗 在 猫 列 表 中 ”这 个 论 
据 真 的 能 够 成 为 它 的 理由 吗 ? 

我 相信 被 称 为 泛 型 的 通用 语言 特性 (并 非 必须 是 其 在 Java 中 的 特定 实现 ) 的 目的 在 于 可 表 
达 性 ， 而 不 仅仅 是 为 了 创建 类 型 安全 的 容器 。 类 型 安全 的 容器 是 能 够 创建 更 通用 代码 这 一 能 力 
所 带 来 的 副作用 。 

因此 ， 即 便 “ 狗 在 猫 列表 中 ”这 个 论据 经 常 被 用 来 证 明 泛 型 是 必要 的 ， 但 是 它 仍旧 是 有 问 
题 的 。 就 像 我 在 本 章 开头 声称 的 ， 我 不 相信 这 就 是 泛 型 这 个 概念 真正 的 含义 。 相 反 ， 泛 型 正如 
其 名 称 所 暗示 的 ， 它 是 一 种 方法 ， 通 过 它 可 以 编写 出 更 “ 泛 化 ”的 代码 ， 这 些 代 码 对 于 它们 能 
够 作用 的 类 型 具有 更 少 的 限制 ， 因 此 单个 的 代码 段 可 以 应 用 到 更 多 的 类 型 上 。 正 如 你 在 本 章 中 
看 到 的 ， 编 写真 正 泛 化 的 “ 持 有 器 ”类 (Java 的 容器 就 是 这 种 类 ) 相当 简单 ， 但 是 编写 出 能 够 
操作 其 泛 型 类 型 的 泛 化 代码 就 需要 额外 的 努力 了 ， 这 些 努 力 需要 类 创建 者 和 类 消费 者 共同 付出 
他 们 必须 理解 适配器 设计 模式 的 概念 和 实现 。 这 些 额外 的 努力 会 增加 使 用 这 种 特性 的 难度 ， 并 
可 能 会 因此 而 使 其 在 某 些 场合 缺乏 可 应 用 性 ， 而 在 这 些 场合 中 ， 它 可 能 会 带 来 附加 的 价值 。 

还 要 注意 到 ， 因 为 泛 型 是 后 来 添加 到 Java 中 ， 而 不 是 从 一 开始 就 设计 到 这 种 语言 中 的 ， 所 
以 某 些 容器 无 法 达到 它们 应 该 具备 的 健壮 性 。 例 如 ， 观 察 一 下 Map ， 在 特定 的 方法 
containsKey(Object key) 和 get(Object key) 中 就 包含 这 类 情况 。 如 果 这 些 类 是 使 用 在 它们 之 前 就 
存在 的 泛 型 设计 的 ， 那 么 这 些 方法 将 会 使 用 参数 化 类 型 而 不 是 Object， 因 此 也 就 可 以 提供 这 些 
泛 型 假设 会 提供 的 编译 期 检查 。 例 如 ， 在 Ct+ 的 map 中 ， 键 的 类 型 总 是 在 编译 期 检查 的 。 

有 一 件 事 很 明显 : 在 一 种 语言 已 经 被 广泛 应 用 之 后 ， 在 其 较 新 的 版 本 中 引入 任何 种 类 的 泛 
型 机 制 ， 都 会 是 一 项 非常 非常 棘手 的 任务 ， 并 且 是 一 项 不 付出 艰辛 就 无 法 完成 的 任务 。 在 C++ 
中 ， 模 版 是 在 其 最 初 的 ISO 版 本 中 就 引入 的 〈 即 便 如 此 ， 也 引发 了 阵痛 ， 因 为 在 第 一 个 标准 C++ 
出 现 之 前 ， 有 很 多 非 模版 版 本 在 使 用 ) ， 因 此 实际 上 模版 一 直 都 是 这 种 语言 的 一 部 分 。 在 Java 中 ， 
泛 型 是 在 这 种 语言 首次 发 布 大 约 10 年 之 后 才 引 入 的 ， 因 此 向 泛 型 迁移 的 问题 特别 多 ， 并 且 对 泛 
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型 的 设计 产生 了 明显 的 影响 。 其 结果 就 是 ， 程 序 员 将 承受 这 些 痛苦 ， 而 这 一 切 都 是 由 于 Java 设 
计 者 在 设计 1.0 版 本 时 所 表现 出 来 的 短视 造成 的 。 当 Java 最 初 被 创建 时 ， 它 的 设计 者 们 当然 了 解 
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C++ 的 模版 ， 他 们 甚至 考虑 将 其 训 括 到 Java 语 言 中 ， 但 是 出 于 这 样 或 那样 的 原因 ， 他 们 决定 将 模 
版 排除 在 外 (其 迹象 就 是 他 们 过 于 匆忙 )。 因 此 ，Java 语 言 和 使 用 它 的 程序 员 都 将 承受 这 些 痛苦 。 
只 有 了 时间 将 会 说 明 Java 的 泛 型 方式 对 这 种 语言 所 造成 的 最 终 影响 。 

某 些 语言 ， 特 别 是 Nice (参见 http://nice.sourceforge.net， 这 种 语言 可 以 产生 Java 字 节 码 ， 并 
可 以 工作 于 现 有 的 Java 类 库 之 上 ) 和 NextGen (参见 http://japan.cs.rice.edu/nextgen) 已 经 融入 了 
更 简洁 、 影 响 更 小 的 方式 ， 来 实现 参数 化 类 型 。 我 们 不 可 能 不 去 想象 这 样 的 语句 将 会 成 为 Java 
的 继任 者 ， 因 为 它们 采用 的 方式 ， 与 C++ 通过 C 来 实现 的 方式 相同 : 按 原样 使 用 它 ， 然 后 对 其 进 
行 改进 。 

15.19.1 进 阶 读物 

关于 泛 型 的 介绍 型 文档 有 Gilad Bracha 所 著 的 《Generics in the Java Programming Language), 
它 位 于 http://java.sun.com/j2se/1.5/pdf/generics-tutorial.pdf。 

Angelika Langer 的 《Java Generics FAQs》 是 一 个 非常 有 用 的 资料 ， 它 位 于 www.angeli- 
kalangercom/GeneiicsFAQ/JavaGenericsFAQ html。 

可 以 在 Torgerson、Emst、Hansen、von der Ahe、Bracha 和 Gafter 所 著 的 《Adding Wildcards 
to the Java Programming Language》 中 找到 更 多 有 关 通 配 符 的 知识 ， 它 位 于 www.jot.fm/issues/ 
issue_2004_12/article5。 

所 选 习题 的 答案 都 可 以 在 名 为 The Thinking in Java Annotated Solution Guide 的 电子 文档 中 找 


到 ， 读 者 可 以 从 wwwMindView.net 处 购买 此 文档 。 


第 16 章 数 组 


在 第 5 章 的 末尾 ， 你 学 习 了 如 何 定义 并 初始 化 一 个 数组 。 

对 数组 的 基本 看 法 是 ， 你 可 以 创建 并 组 装 它们 ， 通 过 使 用 整 型 索引 值 访问 它们 的 元 素 ， 并 
且 它 们 的 尺寸 不 能 改变 。 在 大 多 数 时 候 ， 这 就 是 你 需要 了 解 的 全 部 ， 但 是 有 时 你 需要 在 数组 上 
执行 更 加 复杂 的 操作 ， 并 且 你 可 能 需要 评估 到 底 是 使 用 数组 还 是 更 加 灵活 的 容器 。 本 章 将 向 你 
展示 如 何 更 加 深入 地 思考 数组 。 


16.1 数组 为 什么 特殊 


Java 中 有 大 量 其 他 的 方式 可 以 持 有 对 象 ， 那 么 ， 到 底 是 什么 使 数组 变 得 与 众 不 同 呢 ? 

数组 与 其 他 种 类 的 容器 之 间 的 区 别 有 三 方面 : 效率 、 类 型 和 保存 基本 类 型 的 能 力 。 在 Java 
中 ， 数 组 是 一 种 效率 最 高 的 存储 和 随机 访问 对 象 引 用 序列 的 方式 。 数 组 就 是 一 个 简单 的 线性 序 
列 ， 这 使 得 元 素 访问 非常 快速 。 但 是 为 这 种 速度 所 付出 的 代价 是 数组 对 象 的 大 小 被 固定 ， 并 且 
在 其 生命 周期 中 不 可 改变 。 你 可 能 会 建议 使 用 ArrayList (参见 第 11 章 )， 它 可 以 通过 创建 一 个 新 
实例 ， 然 后 把 旧 实例 中 所 有 的 引用 移 到 新 实例 中 ， 从 而 实现 更 多 空间 的 自动 分 配 。 尽 管 通常 应 
该 首选 ArrayList 而 不 是 数组 ， 但 是 这 种 弹性 需要 开销 ， 因 此 ，ArrayList 的 效率 比 数组 低 很 多 

数组 和 容器 都 可 以 保证 你 不 能 滥用 它们 。 无 论 你 是 使 用 数组 还 是 容器 ， 如 果 越界 ， 都 会 得 
到 一 个 表示 程序 员 错误 的 RuntimeException 异 常 。 

在 泛 型 之 前 ， 其 他 的 容器 类 在 处 理 对 象 时 ， 都 将 它们 视 作 没有 任何 具体 类 型 。 也 就 是 说 ， 
它们 将 这 些 对 象 都 当 作 Java 中 所 有 类 的 根 类 Object 处 理 。 数 组 之 所 以 优 于 泛 型 之 前 的 容器 ， 就 是 
因为 你 可 以 创建 一 个 数组 去 持 有 某 种 具体 类 型 。 这 意味 着 你 可 以 通过 编译 期 检查 ， 来 防止 插入 
错误 类 型 和 抽取 不 当 类 型 。 当 然 ， 无 论 在 编译 时 还 是 运行 时 ，Java 都 会 阻止 你 向 对 象 发 送 不 恰 
当 的 消息 。 所 以 ， 并 不 是 说 哪 种 方法 更 不 安全 ， 只 是 如 果 编 译 时 就 能 够 指出 错误 ， 会 显得 更 加 
优雅 ， 也 减少 了 程序 的 使 用 者 被 异常 吓 着 的 可 能 性 。 

数组 可 以 持 有 基本 类 型 ， 而 泛 型 之 前 的 容器 则 不 能 。 但 是 有 了 泛 型 ， 容 器 就 可 以 指定 并 检 
查 它们 所 持 有 对 象 的 类 型 ， 并 且 有 了 自动 包装 机 制 ， 容 器 看 起 来 还 能 够 持 有 基本 类 型 。 下 面 是 
将 数组 与 泛 型 容器 进行 比较 的 示例 


//: arrays/ContainerComparison.java 
import java.util.*; 
import static net.mindview.util.Print.*; 


class BerylliumSphere { 
private static long counter; 
private final long id = counter++; 
public String toString() { return "Sphere "+ id; } 
} 


public class ContainerComparison { 
public static void main(String{] args) { 
BerylliumSphere{] spheres = new BerylliumSphere[10] ; 
for(int 1 = 6; i < 5; 1++) 
spheres{i] = new BerylliumSphere(); 
print(Arrays.toString(spheres)); 
print (spheres([4]) ; 


747 





= 


434 16 # 





List<BerylliumSphere> sphereList = 
new ArrayList<BerylliumSphere>(); 
for(int i = 0; i < 5; i++) 
sphereList.add(new BerylliumSphere()) ; 
print(sphereList); 
print(sphereList.get(4)); 


int(] integers = { @, 1, 2, 3, 4, 5}; 
print (Arrays. toString(integers)); 
_ print (integers (4]); 


List<Integer> intList = new ArrayList<Integer>( 
Arrays.asList(@, 1, 2, 3, 4, 5)); 

intList.add(97) ; 

print(intList); 

print (intList.get(4)); 


} 
} /* Output: 
[Sphere ©, Sphere 1, Sphere 2, Sphere 3, Sphere 4, null, 
null, null, null, nuti] 
Sphere 4 
{Sphere 5, Sphere 6, Sphere 7, Sphere 8, Sphere 9] 
Sphere 9 
(8, 1, 2, 3, 4, 5) 
4 
[0, 1, 2, 3, 4, 5, 97] 
4 
Wha 


这 两 种 持 有 对 象 的 方式 都 是 类 型 检查 型 的 , 并 且 唯 一 明显 的 差异 就 是 数组 使 用 [] 来 访问 元 素 ， 
而 List 使 用 的 是 add0 和 getO 这 样 的 方法 。 数 组 和 ArrayList 之 间 的 相似 性 是 有 意 设计 的 ， 这 使 得 
从 概念 上 讲 ， 这 两 者 之 间 的 切换 是 很 容易 的 。 但 是 正如 你 在 第 11 章 中 所 看 到 的 ， 容 器 比 数组 明 
显 具 有 更 多 的 功能 。 

随 着 自动 包装 机 制 的 出 现 ， 容 器 已 经 可 以 与 数组 几乎 一 样 方便 地 用 于 基本 类 型 中 了 。 数 组 
硕果 仅 存 的 优点 就 是 效率 。 然 而 ， 如 果 要 解决 更 一 般 化 的 问题 ， 那 数组 就 可 能 会 受到 过 多 的 限 
制 ， 因 此 在 这 些 情形 下 你 还 是 会 使 用 容器 。 


16.2 数组 是 第 一 级 对 象 


无 论 使 用 哪 种 类 型 的 数组 ， 数 组 标识 符 其 实 只 是 一 个 引用 ， 指 向 在 堆 中 创建 的 一 个 真实 对 
象 ， 这 个 (数组 ) 对 象 用 以 保存 指向 其 他 对 象 的 引用 。 可 以 作为 数组 初始 化 语法 的 一 部 分 隐 式 
地 创建 此 对 象 ， 或 者 用 new 表 达 式 显 式 地 创建 。 只 读 成 员 length 是 数组 对 象 的 一 部 分 (事实 上 ， 
这 是 唯一 一 个 可 以 访问 的 字段 或 方法 )， 表 示 此 数组 对 象 可 以 存储 多 少 元 素 。“[]” 语 法 是 访问 数 
组 对 象 唯一 的 方式 。 

下 例 总 结 了 初始 化 数组 的 各 种 方式 ， 以 及 如 何 对 指向 数组 的 引用 赋值 ， 使 之 指向 另 一 个 数 
组 对 象 。 此 例 也 说 明 ， 对 象 数组 和 基本 类 型 数组 在 使 用 上 几乎 是 相同 的 ， 唯 一 的 区 别 就 是 对 象 
数组 保存 的 是 引用 ， 基 本 类 型 数组 直接 保存 基本 类 型 的 值 。 


//: arrays/ArrayOptions.java 

// Initialization & re-assignment of arrays. 
import java.util.*; 

import static net.mindview.util.Print.*; 





public class ArrayOptions { 
public static void main(String{] args) { 
// Arrays of objects: 
BerylliumSphere{] a; // Local uninitialized variable 
BerylliumSphere(} b = new BerylliumSphere(S]; 
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// The references inside the array are 

// automatically initialized to null: 

print("b: " + Arrays.toString(b)): 

BerylliumSphere[] < = new BerylliumSphere[4] ; 

for(int 1= @; i < c.length; i++) 

“if(G[i] == null) // Can test for null reference 

cli] = new BerylliumSphere(); 

// Aggregate initialization: 

BerylliumSphere[] d = { new BerylliumSphere(), 
new BerylliumSphere(), new BerylliumSphere() 


X 
// Dynamic aggregate initialization: 
a = new BerylliumSphere[]{ 
new BerylliumSphere(), new BerylliumSphere(), 


}; 

// (Trailing comma is optional in both cases) 
print(“a. length = "+ a. length); 

print("b. length = " 4 b. length); 

print("c. length ”+ c.length); 
print("d.length = * + d.length): 

a= d; A 

print("a, length = ”+ a. length); 





// Arrays of primitives: 
int[] e; // Null reference 

int[] f = new int[5]; 

// The} primitives inside the array are 

// automatically initialized to zero: 
` print(tf: " + Arrays. toString(f)): 

int[] g = new int[4]; 

for(int i = 0; i < g.length; i++) 

gli] = i*i; 

int[] h = { 11, 47, 93 }; 

// Compile error: variable e not initialized: 
//print("e. length = " + e. length); 





print("f.length = " + f.length); 
print("g.length = " + g. length); 
print("h. length = ”+ h. length); 


e=h; 

print("e.length = ”+ e.length); 
e = new int(}{ 1, 2 }; 
print("e.length = ”+ e. length); 





$ 

/* Output: 

Inuti, null, null, null, nutty 
length = 

length = 
„length = 
length = 
length = 
10, 0, 0 
length = 
length = 
length = 
length = 
length = 
“I~ 


数组 a 是 一 个 尚未 初始 化 的 局 部 变量 ， 在 你 对 它 正确 地 初始 化 之 前 ， 编 译 器 不 允许 用 此 引用 
做 任何 事情 。 数 组 b 初 始 化 为 指向 一 个 BerylliumSphere 引 用 的 数组 ， 但 其 实 并 没有 Beryllium- 
Sphere 对 象 置 人 数组 中 。 然 而 ， 仍 然 可 以 询问 数组 的 大 小 ， 因 为 b 指 向 一 个 合法 的 对 象 。 这 样 
做 有 一 个 小 缺点 ， 你 无 法 知道 在 此 数组 中 确切 地 有 多 少 元 素 ， 因 为 length 只 表示 数组 能 够 容纳 
多 少 元 素 。 也 就 是 说 ，length 是 数组 的 大 小 ， 而 不 是 实际 保存 的 元 素 个 数 。 新 生成 一 个 数组 对 
象 时 ， 其 中 所 有 的 引用 被 自动 初始 化 为 null， 所 以 检查 其 中 的 引用 是 否 为 null， 即 可 知道 数组 的 
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某 个 位 置 是 否 存 有 对 象 。 同 样 ， 基 本 类 型 的 数组 如 果 是 数值 型 的 ， 就 被 自动 初始 化 为 0， 如 果 是 
字符 型 (char) 的 ， 就 被 自动 初始 化 为 (char)O， 如 果 是 布尔 型 《boolean) ， 就 被 自动 初始 化 为 
false, 

数组 c 表 明 ， 数 组 对 象 在 创建 之 后 ， 随 即将 数组 的 各 个 位 置 都 赋值 为 BerylliumSphere 对 象 
数组 4 表明 使 用 “聚集 初始 化 " 语法 创建 数组 对 象 ( 隐 式 地 使 用 new 在 堆 中 创建 ， 就 像 数组 c 一 样 )， 
并 且 以 BerylliumSphere 对 象 将 其 初始 化 的 过 程 ， 这 些 操作 只 用 了 一 条 语句 。 

下 一 个 数组 初始 化 可 以 看 作 是 “动态 的 聚集 初始 化 "。 数 组 d 采 用 的 聚集 初始 化 操作 必须 在 
定义 d 的 位 置 使 用 ， 但 若 使 用 第 二 种 语法 ， 可 以 在 任意 位 置 创建 和 初始 化 数组 对 象 。 例 如 ， 假 设 
方法 hide0 需 要 一 个 BerylliumSphere 对 象 的 数组 作为 输入 参数 。 可 以 如 下 调用 ; 

hide(d); 

但 也 可 以 动态 地 创建 将 要 作为 参数 传递 的 数组 ; 
hide(new BerylliumSphere[]{ new BerylliumSphere(), 
在 许多 情况 下 ， 此 语法 使 得 代码 书写 变 得 更 方便 了 。 

表达 式 : 

aed; 

说 明 如 何 将 指向 某 个 数组 对 象 的 引用 赋 给 另 一 个 数组 对 象 ， 这 与 其 他 类 型 的 对 象 引用 没什么 区 
别 。 现 在 4 与 d 都 指向 堆 中 的 同一 个 数组 对 象 。 

ArraySizejava 的 第 二 部 分 说 明 ， 基 本 类 型 数组 的 工作 方式 与 对 象 数组 一 样 ， 不 过 基本 类 型 
的 数组 直接 存储 基本 类 型 数据 的 值 。 

练习 1: (2) 创建 一 个 受 BerylliumSphere 数 组 作为 参数 的 方法 ， 并 动态 地 创建 参数 去 调用 这 
个 方法 。 证 明 在 本 例 中 普通 的 聚集 数组 初始 化 不 能 奏效 。 去 发 现 总 结 在 哪些 情况 下 ， 普 通 的 聚 
集 初始 化 可 以 起 作用 ， 而 又 在 哪些 情况 下 ， 动 态 聚集 初始 化 显得 多 余 。 


16.3 返回 一 个 数组 


假设 你 要 写 一 个 方法 ， 而 且 希 望 它 返回 的 不 止 一 个 值 ， 而 是 一 组 值 。 这 对 于 C 和 C++ 这 样 的 
语言 来 说 就 有 点 困难 ， 因 为 它们 不 能 返回 一 个 数组 ， 而 只 能 返回 指向 数组 的 指针 。 这 会 造成 一 
些 问题 ， 因 为 它 使 得 控制 数组 的 生命 周期 变 得 很 困难 ， 并 且 容易 造成 内 存 泄漏 。 

在 Java 中 ， 你 只 是 直接 “返回 一 个 数组 "， 而 无 需 担心 要 为 数组 负责 一 -只 要 你 需要 它 ， 它 
就 会 一 直 存在 ， 当 你 使 用 完 后 ， 垃 圾 回收 器 会 清理 掉 它 。 

下 例 演示 如 何 返回 String 型 数组 : 


//: arrays/IceCream. java 
// Returning arrays from methods. 
import java.util.*; 


public class IceCream { 
private static Random rand = new Random(47); 
static final String[] FLAVORS = { 
"Chocolate", "Strawberry", “Vanilla Fudge Swirl", 
“Mint Chip", "Mocha Almond Fudge”, "Rum Raisin", 
“Praline Cream", “Mud Pie" 


oa 
public static String{] flavorSet(int n) { 
if(n > FLAVORS. length) 
throw new ILlegalArgumentException("Set too big"): 
String[] results = new String(n]; 
boolean[] picked = new boolean[FLAVORS. length] ; 
for(int 1 = @; i <n; itt) { 
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int t; 
do 
t = rand.nextInt (FLAVORS. length) ; 
while(picked(t]); 
results[i] = FLAVORS[t]; 
picked[t] = true; 


return results; 


} 
public static void main(String{] args) { 
for(int 1 = 0; i < 7; i++) 
System. out.printIn(Arrays.toString(flavorSet (3))); 

} 
} /* Output: 
[Rum Raisin, Mint Chip, Mocha Almond Fudge] 
[Chocolate, Strawberry, Mocha Almond Fudge] 
(Strawberry, Mint Chip, Mocha Almond Fudge} 
(Rum Raisin, Vanilla Fudge Swirl, Mud Pie] 
(Vanilla Fudge Swirl, Chocolate, Mocha Almond Fudge] 
[Praline Cream, Strawberry, Mocha Almond Fudge] 
[Mocha Almond Fudge, Strawberry, Mint Chip] 
Whim 


方法 flavorSet0 创 建 了 一 个 名 为 results 的 String 数 组 。 此 数组 容量 为 n， 由 传人 方法 的 参数 决 
定 。 然 后 从 数组 FLAVORS 中 随机 选择 元 素 ( 即 “味道 ”)， 存 入 results 数 组 中 ， 它 是 方法 所 最 终 
返回 的 数组 。 返 回 一 个 数组 与 返回 任何 其 他 对 象 (实质 上 是 返回 引用 ) 没什么 区 别 。 数 组 是 在 
flavorSet0 中 被 创建 还 是 在 别 的 地 方 被 创建 ， 这 一 点 并 不 重要 。 当 使 用 完毕 后 ， 垃 圾 回收 器 负责 
清理 数组 ， 而 只 要 还 需要 它 ， 此 数组 就 会 一 直 存在 。 

说 句 题 外 话 ， 注 意 当 favorSet0 随 机 选 洋 各 种 数组 元 素 “ 味 道 ”时 ， 它 确保 不 会 重复 选择 。 
由 一 个 do 循环 不 断 进行 随机 选择 ， 直 到 找 出 一 个 在 数组 picked 中 还 不 存在 的 元 素 。( 当 然 ， 还 会 
比较 String 以 检查 随机 选择 的 元 素 是 否 已 经 在 数组 results 中 。) 如 果 成 功 ， 将 此 元 素 加 入 数组 ， 
然后 查找 下 一 个 (i 递增 )。 

从 输出 中 可 以 看 出 ，flavorSet0 每 次 确实 是 在 随机 选择 “味道 ”。 

练习 2: (1) 编写 一 个 方法 ， 它 接受 一 个 int 参 数 ， 并 返回 一 个 具有 该 尺寸 的 数组 ， 用 
BerylliumSphere 对 象 填 充 该 数组 。 


16.4 多 维 数组 
创建 多 维 数组 很 方便 。 对 于 基本 类 型 的 多 维 数组 ， 可 以 通过 使 用 花 括号 将 每 个 向 量 分 隔 开 ; 


//: arrays/MultidimensionalPrimitiveArray. java 
// Creating multidimensional arrays. 
import java.util.*; 


public class MultidimensionalPrimitiveArray { 
public static void main(String[] args) { 
int a= { 
{1..2, 3, h 
145,6 


hi 
System.out.printLn(Arrays.deepToString(a)): 
} 
} /* Output: 
[[1, 2, 3). (4. 5, 6]] 
Wh 


每 对 花 括号 括 起 来 的 集合 都 会 把 你 带 到 下 一 级 数组 。 
下 面 的 示例 使 用 了 Java SE5 的 Arrays.deepToString0 方 法 ， 它 可 以 将 多 维 数组 转换 为 多 个 
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String， 正 如 从 输出 中 所 看 到 的 那样 。 还 可 以 使 用 new 来 分 配 数组 ， 下 面 的 三 维 数组 就 是 在 new 


表达 式 中 分 配 的 : 


//: arrays/ThreeDWithNew. java 
import java.util.*; 


public class ThreeDWithNew { 
public static void main(String{] args) { 
// 3-D array with fixed length: 
int) []{] a = new int(2)[2) (4): 
System. out.println(Arrays.deepToString(a)) ; 


$ 
} /* Output: 
CCLO, ©, @, O1, (6, ©, 9, 01], [(9, ©, @, OJ, (9, 0, 9, 
911) 
Wh 


你 可 以 看 到 基本 类 型 数组 的 值 在 不 进行 显 式 初始 化 的 情况 下 ， 会 被 自动 初始 化 。 对 象 数组 会 被 


初始 化 为 null。 
数组 中 构成 矩阵 的 每 个 向 量 都 可 以 具有 任意 的 长 度 〈 这 被 称 为 粗 料 数组 ): 


//: arrays/RaggedArray. java 
import java.util.*; 


public class RaggedArray { 
public static void main(String[] args) { 
Random rand = new Random(47) ; 
// 3-D array with varied-length vectors: 
int(}(J{) a = new int{rand.nextInt(7)] (10); 
1 for(int i = 0; i < a.length; i++) { 
ali] = new int[rand.nextint(5)] []; 
| for(int j = 8; j < afi].length; j++) 
alil [j] = new int{rand.nextInt(5)]; 





} 
System.out.println(Arrays. deepToString (a)); 


} 
} /* Output: 
U1, £18}, (6), 10, ©, ©, 0)), IN. 8. 0). [9, 0J). (18, 
. 8]], (18, @, 6], [9, 9, 9], (0), [1]. 





tiel, 0, £613) 
HE 


第 一 个 new 创 建 了 数组 ， 其 第 一 维 的 长 度 是 由 随机 数 确定 的 ， 其 他 维 的 长 度 则 没有 定义 。 位 于 
for 循 环 内 的 第 二 个 new 则 会 决定 第 二 维 的 长 度 ， 直 到 碰 到 第 三 个 new， 第 三 维 的 长 度 才 得 以 


确定 。 


可 以 用 类 似 的 方式 处 理 非 基本 类 型 的 对 象 数组 。 下 面 ， 你 可 以 看 到 如 何 用 花 括号 把 多 个 new 


表达 式 组 织 到 一 起 : 


//: arrays/MultidimensionalObjectArrays.java 
import java.util.*; 


public class MultidimensionalObjectArrays { 
Public static void main(String{] args) { 
BerylliumSphere(][] spheres = { 
{ new BerylliumSphere(), new BerylliumSphere() }, 
{ new BerylliumSphere(), new BerylliumSphere(), 
new BerylliumSphere(), new BerylliumSphere() }, 
{ new BerylliumSphere(), new BerylliumSphere(), 
new BerylliumSphere(), new BerylliumSphere(), 
new BerylliumSphere(), new BerylliumSphere(), 
new BerylliumSphere(), new BerylliumSphere() }, 





System. out. printin(Arrays .deepToString(spheres)) ; 





} 
} /* Output: 
({Sphere 6, Sphere 1], [Sphere 2, Sphere 3, Sphere 4, 
Sphere 5], [Sphere 6, Sphere 7, Sphere 8, Sphere 9, Sphere 
10, Sphere 11, Sphere 12, Sphere 13]] 
When 


你 可 以 看 到 spheres 是 另外 一 个 粗糙 数组 ， 其 每 一 个 对 象 列 表 的 长 度 都 是 不 同 的 。 
自动 包装 机 制 对 数组 初始 化 器 也 起 作用 : 


//: arrays/AutoboxingArrays. java 
import java.util.*; 


public class AutoboxingArrays { 
public static void main(String[] args) { 
Integer[][] a = { // Autoboxing: 
{ 1,2, 3, 4, 5. 6.7, 8, 9, 18}, 
{ 21, 22, 23, 24, 25, 26, 27, 28, 29, 30}, 
{ 51, 52, 53, 54, 55, 56, 57, 58, 59, 60 }, 
{ 71, 72, 73, 74, 75, 76, 77, 78, 79, 80 }, 


Ji 
System.out.println(Arrays.deepToString(a)); 


} 
} /* Output: 

tl, 2, 3, 4, 5, 6, 7, 8, 9, 10), [21, 22, 23, 24, 25, 2 
27, 28, 29, 30), [51, 52, 53, 54, 55, 56, 57, 58, 59, ear, 
(71, 72, 73, 74, 75, 76, 77, 78, 79, 80]) 

“H 


下 面 的 示例 展示 了 可 以 如 何 逐 个 地 、 部 分 地 构建 一 个 非 基 本 类 型 的 对 象 数组 ; 


//: arrays/AssemblingMultidimensionalArrays. java 
// Creating multidimensional arrays. 
import java.util.*; 





public class AssemblingHultidimensionalArrays { 
public static void main(String[] args) { 
Integer {]{] a: 
a = new Integer [3] []; 
for(int i = 0; 1 < a.length; i++) { 
afi] = new Integer [3]: 
for(int j = 9; j < ali].length; j++) 
ali}{j] = * j; // Autoboxing 


} 
System. out.printin(Arrays.deepToString(a)); 


$ 
} /* Output: 
{[9, @, 6}, [9, 1, 2]. [@0, 2, 4]] 
s~ 


这 只 是 为 了 使 置 于 Integer 中 的 值 变 得 有 些 意思 。 
Arrays.deepToString0 方 法 对 基本 类 型 数组 和 对 象 数组 都 起 作用 : 


11: arcays/MultiDimWrapperArray. java 
// Multidimensional arrays of "wrapper" objects. 
import java.util.*; 


public class MultiDimWrapperArray { 
public static void main(String[] args) { 
Integer[][] al = { // Autoboxing 





£4,.25-30 3, 
{4, 5, 6, } 

Double[][][] a2 = { // Autoboxing 
{ (2.1, 2.23, (3.3 4) }. 
{ (5.5, 6.6}, (7.7, 8.8} }, 
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{ (9.9, 1.2}, (2.3, 3.4)), 


}; 
String[][] a3 = { 
{ "The", "Quick", "Sly", "Fox" }, 
{ "Jumps "Over" }, 
{ "The", "Lazy", "Brown", "Dog", "and". “friend” }, 








}; 
System.out.println("al: "+ Arrays.deepToString(al)); 
System.out.printin("a2: " + Arrays.deepToString(a2)); 
System.out.printin("a3: ”+ Arrays.deepToString(a3)); 


} 
} /* Output: 

al: [[1, 2, 3]. [4, 5, 61) 

a2: (11.1, 2.2), [3.3, 4.4]], [[5.5, 6.6], (7.7, 8.8]], 
(19.9, 1.2], (2.3, 3.4]]] 

a3: [[The, Quick, Sly, Fox], [Jumped, Over}, [The, Lazy, 
Brown, Dog, and, friend]] 

Wh 


在 Integer 和 Double 数 组 中 ，Java SE5 的 自动 包装 机 制 再 次 为 我 们 创建 了 包装 器 对 象 。 


练习 3: (4) 编写 一 个 方法 ， 能 够 产生 二 维 双 精度 型 数组 并 加 以 初始 化 。 数 组 的 容量 由 方法 
的 形式 参数 决定 ， 其 初 值 必 须 落 在 另外 两 个 形式 参数 所 指定 的 区 间 之 内 。 编 写 第 二 个 方法 ， 打 
758) 印 出 第 一 个 方法 所 产生 的 数组 。 在 main0 中 通过 产生 不 同 容量 的 数组 并 打印 其 内 容 来 测试 这 两 














个 方法 。 
练习 4，(2) 重复 前 一 个 练习 ， 但 改 为 三 维 数组 。 
练习 5: (1) 证 明基 本 类 型 的 多 维 数组 会 自动 被 初始 化 为 null。 


练习 6: (1) 编写 一 个 方法 ， 它 接受 两 个 表示 二 维 数组 尺寸 的 int 参 数 。 这 个 方法 应 该 这 两 个 


' 根据 尺寸 参数 ， 创 建 并 填充 一 个 BerylliumSphere 二 维 数组 。 
| 练习 7，(]) 重复 前 一 个 练习 ， 但 改 为 三 维 数组 。 


' 16.5 数组 与 泛 型 


通常 数组 与 泛 型 不 能 很 好 地 结合 。 你 不 能 实例 化 具有 参数 化 类 型 的 数组 : 


Peel<Banana>{] peels = new Peel<Banana>[10}; // Illegal 


擦 除 会 移 除 参数 类 型 信息 ， 而 数组 必须 知道 它们 所 持 有 的 确切 类 型 ， 以 强制 保证 类 型 安全 。 


但 是 ， 你 可 以 参数 化 数组 本 身 的 类 型 : 


/1/: arrays/Parameter izedArrayType. java 


class ClassParameter<T> { 
public TI[] f(T[] arg) { return arg; } 
$ 


class MethodParameter { 
public static <T> T[] f(T[] arg) { return arg; } 
} 


public class ParameterizedArrayType { 
public static void main(String{] args) { 
Integer{] ints = { 1, 2, 3, 4, 5}; 
Double[] doubles = { 1.1, 2.2, 3.3, 4.4, 5.5 }: 
Integer (] ints2 = 
new ClassParameter<Integer>().f (ints); 
Double{] doubles2 = 
759| new ClassParameter<Double>() .f (doubles) ; 
ints2 = MethodParameter .f (ints); 
doubles2 = MethodParameter . f (doubles); 
} 
} 1:~ 
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注意 ， 使 用 参数 化 方法 而 不 使 用 参数 化 类 的 方便 之 处 在 于 : 你 不 必 为 需要 应 用 的 每 种 不 同 
的 类 型 都 使 用 一 个 参数 去 实例 化 这 个 类 ， 并 且 你 可 以 将 其 定义 为 静态 的 。 当 然 ， 你 不 能 总 是 选 
择 使 用 参数 化 方法 而 不 是 参数 化 类 ， 但 是 它 应 该 成 为 首选 。 

正如 上 例 所 证 明 的 那样 ， 不 能 创建 泛 型 数组 这 一 说 法 并 不 十 分 准确 。 诚 然 ， 编 译 器 确实 不 
让 你 实例 化 泛 型 数组 ， 但 是 ， 它 允许 你 创建 对 这 种 数组 的 引用 。 例 如 : 

List<String>(] 1s; 

这 条 语句 可 以 顺利 地 通过 编译 器 而 不 报 任何 错误 。 而 且 ， 尽 管 你 不 能 创建 实际 的 持 有 泛 型 
的 数组 对 象 ， 但 是 你 可 以 创建 非 泛 型 的 数组 ， 然 后 将 其 转型 : 


//: arrays/ArrayOfGenerics.java 
// It is possible to create arrays of generics. 
import java.util.*; 


public class ArrayOfGenerics { 

@SuppressWarnings ("unchecked") 

public static void main(String] args) { 
ListsString>[] 1s; 
List] la = new List[19]; 
Is = (ListxString>[])1a; // "Unchecked" warning 
1s [8] = new ArrayList<String>(); 
// Compile-time checking produces an error: 
/AL 1s[1] = new ArrayList<Integer>(); 


// The problem: List<String> is a subtype of Object 
Object[] objects = ls; // So assignment is OK 

// Compiles and runs without complaint: 

objects[1] = new ArrayList<Integer>(); 


// However, if your needs are straightforward it is 
// possible to create an array of generics, albeit 
// with an “unchecked” warning: 
List<BerylliumSphere>[} spheres = 
(List<Bery1liumSphere>{})new List [16] ; 
for(int i = @; i < spheres.length; i++) 
spheres{i] = new ArrayList<BerylliumSphere>(); 


) 

} i~ 

一 旦 拥有 了 对 List<String>[] 的 引用 ， 你 就 会 看 到 你 将 得 到 某 些 编译 器 检查 。 问 题 是 数组 是 
协 变 类 型 的 ， 因 此 List<String>[] 也 是 一 个 Object[]， 并 且 你 可 以 利用 这 一 点 ， 将 一 个 
ArrayList<Integer> 赋 值 到 你 的 数组 中 ， 而 不 会 有 任何 编译 期 或 运行 时 错误 。 

如 果 你 知道 将 来 不 会 向 上 转型 ， 并 且 需 求 也 相对 比较 简单 ， 那 么 你 仍旧 可 以 创建 泛 型 数组 ， 
它 可 以 提供 基本 的 编译 期 类 型 检查 。 但 是 ， 事实 上 ， 泛 型 容器 总 是 比 泛 型 数据 更 好 的 选择 。 

一 般 而 言 ， 你 会 发 现 泛 型 在 类 或 方法 的 边界 处 很 有 效 ， 而 在 类 或 方法 的 内 部 ， 擦 除 通常 会 
使 泛 型 变 得 不 适用 。 例 如 ， 你 不 能 创建 泛 型 数组 


//: arrays/ArrayOfGenericType.java 
// Arrays of generic types won't compile. 


public class ArrayOfGenericType<T> { 
TU array; // OK 
@SuppressWarnings ("unchecked") 
public ArrayOfGenericType(int size) { 
//! array = new T[size]; // Illegal 
array = (T[))new Object[size]; // “unchecked” Warning 


} 

/11 Thlegal: 

//! public <U> U[] makeArray() { return new U[10]; } 
} Uhm 
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擦 除 再 次 成 为 了 障碍 一 一 本 例 试图 创建 的 类 型 已 被 擦 除 ， 因 而 是 类 型 未 知 的 数组 。 注 意 ， 
你 可 以 创建 Object 数组 ， 然 后 将 其 转型 ， 但 是 如 果 没 有 @SuppressWarnings 注 解 ， 你 将 在 编译 
期 得 到 一 个 “不 受 检查 ”的 警告 消息 ， 因 为 这 个 数组 没有 真正 持 有 或 动态 检查 类 型 T。 也 就 是 说 ， 
如 果 我 创建 一 个 String[]，Java 在 编译 期 和 运行 时 都 会 强制 要 求 我 只 能 将 String 对 象 置 于 该 数组 
中 。 但 是 ， 如 果 创 建 的 是 Object]， 那 么 我 就 可 以 将 除 基本 类 型 之 外 的 任何 对 象 置 于 该 数组 中 。 

练习 8: (1) 证 明 前 一 段 话 中 的 断言 。 

练习 9: (3) 创建 Peel<Banana> 所 必需 的 类 ， 并 展示 编译 器 不 会 接受 它 。 使 用 ArrayList 来 改 


正 此 问题 。 
练习 10: (2) 修改 ArrayOfGenerics.java， 在 其 中 使 用 容器 而 不 是 数组 。 展 示 你 可 以 根除 编 


译 期 警告 信息 。 
16.6 创建 测试 数据 


通常 ， 在 试验 数组 和 程序 时 ， 能 够 很 方便 地 生成 填充 了 测试 数据 的 数组 ， 将 会 很 有 帮助 。 
本 节 介绍 的 工具 就 可 以 用 数值 或 对 象 来 填充 数组 。 
16.6.1 Arrays.fill() 

Java 标 准 类 库 Arrays 有 一 个 作用 十 分 有 限 的 fi10 方 法 ， 只 能 用 同一 个 值 填充 各 个 位 置 ， 而 针 
对 对 象 而 言 ， 就 是 复制 同一 个 引用 进行 填充 。 下 面 是 一 个 示例 : 


11: arrays/FillingArrays.java 
77 Using Arrays.f111() 

import java.util.*; 

import static net.mindview.util.Print.*; 


public class FillingArrays { 

public static void main(String(] args) ( 
int size = 6; 
boolean{] al = new boolean{size] ; 
bytel] a2 = new byte[size] ; 
char[] a3 = new char(size]; 
short(] a4 = new short[size]; 
int[] a5 = new int[size]: 
long[] a6 = new long[size]; 
float{] a7 = new float[size]; 
doubte[] a8 = new doubte[size]; 
Stringl} a9 = new String[size]; 
Arrays.fill(al, true); 
print("al = ”+ Arrays. toString(al)); 
Arrays.fill(a2, (byte)11); 
print("a2 = " + Arrays. toString(a2)); 
Arrays.fill(a3, 'x'); 
print("a3 = ”+ Arrays. toString(a3)); 
Arrays.fill(a4, (short)17); 
print("a4 = " + Arrays. toString(a4)); 
Arrays. fill(aS, 19); 
print("aS = ”+ Arrays.toString(a5)); 
Arrays. fill(a6, 23); 
print("a6 = " + Arrays. toString(a6)); 
Arrays. fill(a7, 29); 
print("a7 = " + Arrays. toString(a7)); 
Arrays. fill(a8, 47); 
print("a8 = * + Arrays. toString(a8)); 
Arrays. fili (a9, "Hello"); 
print("a9 = " + Arrays. toString(a9)); 
// Manipulating ranges: 
Arrays. fill(a9, 3, 5, World"); 
print("a9 = ”+ Arrays. toString(a9)); 
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} 
} /* Output: 
al = [true, true, true, true, true, true] 
a2 = [11, 12, 11, 11, 41, 11) 
a3 = [x, x, x, x, x, x] 
a4 = (17, 17, 17, 17, 17, 17] 
a5 = [19, 19, 19, 19, 19, 19] 
a6 = (23, 23, 23, 23, 23, 23] 
a7 = [29.0, 29.0, 29.0, 29.0, 29.0, 29.0) 
a8 = (47.0, 47.0, 47.0, 47.0, 47.0, 47.0] 
a9 = [Hello, Hello, Hello, Hello, Hello, Hello) 
a9 = [Hello, Hello, Hello, World, World, Hello} 
Wha 


使 用 Arrays.fil10 可 以 填充 整个 数组 ， 或 者 像 最 后 两 条 语句 所 示 ， 只 填充 数组 的 某 个 区 域 。 
但 是 由 于 只 能 用 单一 的 数值 来 调用 Arrays-fil0， 因 此 所 产生 的 结果 并 非特 别 有 用 。 


16.6.2 数据 生成 器 

为 了 以 灵活 的 方式 创建 更 有 意义 的 数组 ， 我 们 将 使 用 在 第 15 章 中 引入 的 Generator 概 念 。 如 
果菜 个 工具 使 用 了 Generator， 那 么 你 就 可 以 通过 选择 Generator 的 类 型 来 创建 任何 类 型 的 数据 
(这 是 策略 设计 模式 的 一 个 实例 一 -每 个 不 同 的 Generator 都 表示 一 个 不 同 的 策略 9 )。 

本 节 将 提供 一 些 Generator， 并 且 ， 就 像 之 前 看 到 的 ， 你 还 可 以 很 容易 地 定义 自己 的 
Generator。 

首先 给 出 的 是 可 以 用 于 所 有 基本 类 型 的 包装 器 类 型 ， 以 及 String 类 型 的 最 基本 的 计数 生成 器 
集合 。 这 些 生成 器 类 都 嵌 套 在 CountingGenerator 类 中 ， 从 而 使 得 它们 能 够 使 用 与 所 要 生成 的 对 
象 类 型 相同 的 名 字 。 例 如 ， 创 建 Integer 对 象 的 生成 器 可 以 通过 表达 式 new CountingGenerator. 
JInteger0 来 创建 ， 


//: net/mindview/uti1/CountingGenerator. java 
// Simple generator implementations. 
package net.mindview.util; 


public class CountingGenerator { 
public static class 
Boolean implements Generator<java.1ang.Boolean> { 
private boolean value = false; 
public java.lang.Boolean next() { 
value = !value; // Just flips back and forth 
return value; 


public static class 
Byte implements Generator<java.lang.Byte> { 
private byte value = 0; 
public java.lang.Byte next() { return value++; } 


$ 
static char[] chars = ("abcdefghijkimnopqrstuvwxyz" + 
"ABCDEFGHI JKLMNOPQRSTUVWXYZ").. toCharArray () ; 
public static class 
Character implements Generator<java.lang.Character> { 
int index = -1; 
public java. lang.Character next() { 
‘index = (index + 1) % chars. length; 
return chars{index] ; 


} 


} 
public static class 


O 尽管 在 这 里 事情 变 得 有 点 含糊 ， 可 能 在 此 会 产生 争议 ,认为 Generator 应 用 了 命令 模式 ， 但 是 我 认为 ， 实 际 的 
任务 是 填充 数组 ， 而 Generator 只 实现 了 该 任务 的 一 部 分 ， 因 此 它 更 像 是 策略 而 不 是 命令 。 
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String implements Generator<java.lang.String> { 
private int length = 7; 
Generator<java.lang.Character> cg = new Character(): 
public String() (} 
public String(int length) { this.length = length; } 
public java.lang.String next() { 

char[] buf = new char [length] ; 

for(int i = 0; i < length: i++) 
bufli] = cg.next(); 

return new java. lang. String(buf); 


2 





à 

public static class 

Short implements Generator<java.lang.Short> { 
private short value = 0; 
public java.lang.Short next() { return value++; } 

} 

public static class 

Integer implements Generator<java.lang.Integer> { 
private int value = 0; 
public java. lang. Integer next() { return value++; } 


public static class 
Long implements Generator<java.lang.Long> { 
private long value = 0; 
public java.lang.Long next() { return value++; } 
} 
public static class 
Float implements Generator<java.lang.Float> { 
private float value = 0; 
public java.lang.Float next() { 
float result = value; 
value += 1.0; 
return result: 


} 


public static class 
Double implements Generator<java.lang.Double> { 
private doubte value = 0.0; 
public java.lang.Double next() { 
double result = value; 


value += 1. 
return result; 


} 
} 

EE 

上 面 的 每 个 类 都 实现 了 某 种 意义 的 “计数 "。 在 CountingGenerator.Character 中 ， 计 数 只 是 
不 断 地 重复 大 写 和 小 写字 母 ， CountingGenerator.String 类 使 用 CountingGenerator.Character 来 
填充 一 个 字符 数组 ， 该 数组 将 被 转换 为 String， 数 组 的 尺寸 取决 于 构造 器 参数 。 请 注意 ， 
CountingGenerator.String 使 用 的 是 基本 的 Generator<java.lang.Character>， 而 不 是 具体 的 对 
CountingGenerator.Character 的 引用 。 稍 后 ， 我 们 可 以 替换 这 个 生成 器 ， 以 生成 Random- 
Generatorjava 中 的 RandomGeneratorString。 

下 面 是 一 个 测试 工具 ， 针 对 左 套 的 Generator 这 一 惯用 法 ， 因 为 使 用 了 反射 所 以 这 个 工具 可 
以 遵循 下 面 的 形式 来 测试 Generator 的 任何 集合 


//: arrays/GeneratorsTest.java 
import net.mindview.util.*: 





public class GeneratorsTest { 
public static int size = 10; 
public static void test(Class<?> surroundingClass) { 
for(Class<?> type : surroundingClass.getClasses()) 
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System.out.print(type.getSimpleName() + * 
try { 
Generator<?> g = (Generator<?>)type.newInstance(); 
for(int i = @; i < size; i++) 
System.out.printf(g.next() +" "); 
System.out.printin(); 
} catch(Exception e) { 
throw new RuntimeException(e); 
} 
} 
} 
public static void main(String(] args) { 
test (Count ingGenerator .class); 


} 


} /* Output: 
Double: 0.0 1.0 2.0 3.0 4.0 5.0 6.0 7.0 8.6 9.0 
Float: 0.0 1.0 2.0 3.0 4.6 5.0 6.0 7.0 8.0 9.0 [766] 





Long: 0123456789 


Integer: 0123456789 
Short: 0123456789 
String: abcdefg hijktmn opqrstu vwxyzAB CDEFGHI JKLMNOP 


QRSTUVW XYZabcd efghijk tmnopgr 
Character: abcdefghij 

Byte: 0123456789 

Boolean: true false true false true false true false true 
false 

i~ 


这 里 假设 待 测 类 包含 一 组 做 套 的 Generator 对 象 ， 其 中 每 个 都 有 一 个 默认 构造 器 (无 参 构造 器 )。 
反射 方法 getClasses0 可 以 生成 所 有 的 做 套 类 ， 而 test0 方 法 可 以 为 这 些 生成 器 中 的 每 一 个 都 创建 
一 个 实例 ， 然 后 打印 通过 调用 10 次 next0 方 法 而 产生 的 结果 。 

下 面 是 一 组 使 用 随机 数 生成 器 的 Generator。 因 为 Random 构 造 器 使 用 常量 进行 初始 化 ， 所 
以 ， 每 次 用 这 些 Generator 中 的 一 个 来 运行 程序 时 ， 所 产生 的 输出 都 是 可 重复 的 : 


//: net/mindview/util/RandomGenerator .java 
// Generators that produce random values. 
package net.mindview.util; 

import java.util.* 





public class RandomGenerator { 
private static Random r = new Random(47); 
public static class 
Boolean implements Generator<java.lang.Boolean> { 
public java.lang.Boolean next() { 
return r.nextBoolean(); 
} 
} 
public static class ， 
Byte implements Generator<java. lang.Byte> { 
public java.lang.Byte next() { 
return (byte)r.nextInt(); 
} a 
} 
public static class 
Character implements Generator<java.lang.Character> { 
public java.lang.Character next() { 
return CountingGenerator .chars{ 767 
r mextInt (CountingGenerator .chars.length)) ; 














$ 
} 
public static class 
String extends CountingGenerator.String { 
// Plug in the random Character generator: 
{ cg = new Character(); } // Instance initializer 
public String() {} 








446 F1O¥ 





public String(int length) { super(length): } 
} 
public static class 
Short implements Generator<java.lang.Short> { 
public java.lang.Short next() { 
return (short)r.nextInt(); 
} 
} 
public static class 
Integer implements Generator<java.lang.Integer> { 
private int mod = 10000; 
public Integer() (} 
public Integer(int modulo) { mod 
public java.lang.Integer next() { 
return r.nextInt (mod) ; 


modulo; } 





} 

} 

public static class 

Long implements Generator<java.lang.Long> { 
private int mod = 10000; 
public Long() {} 
public Long(int modulo) { mod = modulo; } 
public java.lang.tong next() { 

return new java.lang.Long(r.nextInt(mod)); 

} 


} 
public static class 
Float implements Generator<java.lang.Float> { 
public java. lang.Float next() { 
JJ Trim all but the first two decimal places: 
int trimmed = Math. round(r.nextFloat() * 160); 
return ((float)trimmed) / 108; 
} 
} 
public static class 
Double implements Generator<java. lang.Double> { 
public java.lang.Double next() { 
long trimmed = Math. round(r.nextDouble() * 100); 
return ((double)trimmed) / 100; 
} 
} 
dh 


你 可 以 看 到 ，RandomGenerator.String 继 承 自 CountingGenerator.String， 并 且 只 是 插入 了 新 的 
Character 生 成 器 。 

为 了 不 生成 过 大 的 数字 ，RandomGenerator.Integer 默 认 使 用 的 模 数 为 10 000， 但 是 重 载 的 
构造 器 允许 你 选择 更 小 的 值 。 同 样 的 方式 也 应 用 到 了 RandomGenerator.Long 上 。 对 于 Float 和 
Double 生 成 器 ， 小 数 点 之 后 的 数字 被 截 掉 了 。 

我 们 复 用 GeneratorTest 来 测试 RandomGenerator: 


//: arrays/RandomGeneratorsTest.java 
import net.mindview.util.*; 


public class RandomGeneratorsTest { 

public static void main(String[] args) { 

GeneratorsTest. test (RandomGenerator .class); 

} 
} /* Output: 
Double: @.73 0.53 0.16 0.19 @.52 0.27 0.26 0.05 0.8 8.76 
Float: 0.53 0.16 0.53 @.4 6.49 6.25 0.8 6.11 0.02 6.8 
Long: 7674 8864 8950 7826 4322 896 8033 2984 2344 5810 
Integer: 8303 3141 7138 6012 9966 8689 7185 6992 5746 3976 
Short: 3358 20592 284 26791 12834 -8092 13656 29324 -1423 
5327 
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String: bkInaMe sbtWHkj UrUkZPg wsqPzDy CyRFJQA HxxHvHq 
XumcXZJ oogoYWM NvqeuTp nXsgqia 

Character: x x EAJJmzMs 

Byte: -69 -17 55 -14 -5 115 39 -37 79 115 

Boolean: false true false false true true true true true 
true 

:~ 


你 可 以 通过 修改 public 的 GeneratorTest.size 的 值 ， 来 改变 所 产生 的 数值 数量 。 
16.6.3 从 Generator 中 创建 数组 

为 了 接收 Generator 并 产生 数组 ， 我 们 需要 两 个 转换 工具 。 第 一 个 工具 使 用 任意 的 
Generator 来 产生 Object 子 类 型 的 数组 。 为 了 处 理 基 本 类 型 ， 第 二 个 工具 接收 任意 基本 类 型 的 包 
装 器 类 型 数组 ， 并 产生 相应 的 基本 类 型 数组 。 

第 一 个 工具 有 两 种 选项 ， 并 由 重 载 的 静态 方法 array0 来 表示 。 该 方法 的 第 一 个 版 本 接收 一 
个 已 有 的 数组 ， 并 使 用 某 个 Generator 来 填充 它 ， 而 第 二 个 版 本 接收 一 个 Class 对 象 、 一 个 
Generator 和 所 需 的 元 素数 量 ， 然 后 创建 一 个 新 数组 ， 并 使 用 所 接收 的 Generator 来 填充 它 。 注 
意 ， 这 个 工具 只 能 产生 Object 子 类 型 的 数组 ， 而 不 能 产生 基本 类 型 数组 : 


//: net/mindview/util/Generated. java 
package net.mindview.util; 
import java.util.*; 


public class Generated { 
// Fill an existing array: 
public static <T> T[] array(T[] a, Generator<T> gen) { 
return new CollectionData<T>(gen, a.length).toArray(a); 
} 
// Create a new array: 
@SuppressWarnings("unchecked") 
public static <T> TL] array(Class<T> type, 
Generator<T> gen, int size) { 
TU a = 
(T[])java. lang. reflect.Array.newInstance(type, size);` 
return new CollectionDatacT>(gen, size).toArray(a); 


} 
} Mi~ 


CollectionData 类 将 在 第 17 章 中 定义 ， 它 将 创建 一 个 Collection 对 象 ， 该 对 象 中 所 填充 的 元 素 
是 由 生成 器 gen 产 生 的 ， 而 元 素 的 数量 则 由 构造 器 的 第 二 个 参数 确定 。 所 有 的 Collection 子 类 型 
都 拥有 toArray0 方 法 ， 该 方法 将 使 用 Collection 中 的 元 素来 填充 参数 数组 。 

第 二 个 方法 使 用 反射 来 动态 创建 具有 恰当 类 型 和 数量 的 新 数组 ， 然 后 使 用 与 第 一 个 方法 相 
同 的 技术 来 填充 该 数组 。 

我 们 可 以 使 用 在 前 一 节 中 定义 的 CountingGenerator 类 中 的 某 个 生成 器 来 测试 Generated: 


//: arrays/TestGenerated. java 
import java.util.*; 
import net.mindview.util.*; 


public class TestGenerated { 

public static void main(String[] args) { 
Integer{] a= (9, 8 7, 6 }; 
System. out .print1n (Arrays. toString(a)) ; 
a = Generated. array(a,new Count ingGenerator.Integer()); 
System. out.printin (Arrays. toString(a)): 
Integer[] b = Generated. array(Integer.class, 

new CountingGenerator.Integer(), 15); 

System.out.printin(Arrays.toString(b)): 


d 
} 7* Output: 








769} 

















448 钊 16 章 








(772| 











(9. 8, 7, 6) 

[9. 2, 2. 3) 

(0, 1, 2, 3, 4, 5, 6, 7, 8, 9, 19, 11, 12, 13, 14) 
Wh 


即使 数组 a 被 初始 化 过 ， 其 中 的 那些 值 也 在 将 其 传递 给 Generated.array0 之 后 被 复写 了 ， 
为 这 个 方法 会 替换 这 些 值 (但 是 会 保证 原 数组 的 正确 性 )。b 的 初始 化 展示 了 如 何 从 无 到 有 地 创 
建 填充 了 元 素 的 数组 。 

泛 型 不 能 用 于 基本 类 型 ， 而 我 们 确实 想 用 生成 器 来 填充 基本 类 型 数组 。 为 了 解决 这 个 问题 ， 
我 们 创建 了 一 个 转换 器 ， 它 可 以 接收 任意 的 包装 器 对 象 数组 ， 并 将 其 转换 为 相应 的 基本 类 型 数 
组 。 如 果 没 有 这 个 工具 ， 我 们 就 必须 为 所 有 的 基本 类 型 创建 特殊 的 生成 器 。 


//: net/mindview/util/ConvertTo. java 
package net.mindview.util; 





public class ConvertTo { 
public static boolean{] primitive(Boolean{} in) { 
boolean[] result = new boolean[in. length] ; 
for(int i = @; i < in.length: i++) 
result[i] = infil; // Autounboxing 
return result; 
} 
public static char[] primitive(Character[] in) { 
char[] result = new char[in. length); 
for(int i = 0; i < in.length; i++) 
result[i] = infil; n 
return result; 


} 

public static byte[] primitive(Byte[] in) { 
byte[] result = new byte[in. length]; 
for(int 1 =@; 1 < in.length; i++) 

result(i] = inti); 
return result; 

} 

Public static short[] primitive(Short{] in) { 
short{] result = new short{in. length]; 
for(int 1 = @; i < in.length; i++) 

result(i] = in[i]; 
return result; 


} 
public static int[] primitive(Integer[] in) { 
int(] result = new int[in. length]; 
for(int 1 = @; i < in.length; i++) 
result[i] = in[i]; 
return result; 


} 
public static long[] primitive(Long[] in) { 
Long[] result = new longlin. length); 
for(int i = 0; 1 < in.length; i++) 
result[i] = infi]; 
return result; 


F 
public static float[] primitive(Float(] in) { 
float[] result = new float[in. length); 
for(int 1 =@; 1 < in.length; i++) 
result[i] = in[i]; 
return result; 





public static double[] primitive(Double[] in) { 
double[] result = new double[in. length] ; 
for(int i = 8; i < in.length; i++) 
result(i] = in[i]; 
return result 








} 
} H~ 
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Primitive( 方 法 的 每 个 版 本 都 可 以 创建 适当 的 具有 恰当 长 度 的 基本 类 型 数组 ， 然 后 向 其 中 复 
制 包装 器 类 型 数组 中 的 元 素 。 注 意 ， 在 下 面 的 表达 式 中 会 进行 自动 拆 包 : 

result{i] = infil: 

下 面 的 示例 展示 了 如 何 将 ConvertTp 应 用 于 两 个 版 本 的 Generated.array0 上 : 


11: arrays/PrimitiveConversionDemonstration.java 
import java.util.*; 
import net.mindview.util.*: 





public class PrimitiveConversionDemonstration { 
public static void main(String[] args) { 
Integer[] a = Generated. array(Integer .class, 
new CountingGenerator.Integer(). 15); 
int[] b = ConvertTo.primitive(a); 
System. out.printin (Arrays. toString(b)); 
boolean{] c = ConvertTo. primitive( 
Generated. array (Boolean.class, 
new Count ingGenerator.Boolean(), 7)); 
System. out.printIn (Arrays. toString(c)): 
} 
} /* Output: 
(0, 1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 11, 12, 13, 14) 
Itrue, false, true, false, true, false, true) 
“i 


最 后 ， 下 面 的 程序 将 使 用 RandomGenerator 中 的 类 来 测试 这 些 数组 生成 工具 : 


/1/: arrays/TestArrayGeneration. java 
// Test the tools that use generators to fill arrays. 
import java.util.* 
import net.mindview.util.*; 

import static net.mindview.util.Print.*; 








public class TestArrayGeneration { 
public static void main(String[] args) { 
int size = 6; 
boolean[] al = ConvertTo.primitive(Generated.array( 
Boolean.class, new RandomGenerator .Boolean(), size)); 
print("al = " + Arrays. toString(al)); 
bytel] a2 = ConvertTo.primitive(Generated.array( 
Byte.class, new RandomGenerator.Byte(), size)); 
print("a2 = " + Arrays.toString(a2)); i 
char{] a3 = ConvertTo.primitive(Generated.array( 
Character class, 
new RandomGenerator.Character(), size)): 
print("a3 = " + Arrays. toString(a3)); 
Short[] a4 = ConvertTo. primitive (Generated. array( 
Short.class, new RandomGenerator.Short(), size)); 
print("a4 = ”+ Arrays. toString(a4)); 
int{] a5 = ConvertTo. primitive (Generated. array( 
Integer.class, new RandomGenerator.Integer(), size)); 
print("aS = ”+ Arrays. toString(a5)) ; 
long[] a6 = ConvertTo. primitive (Generated. array( 
Long.class, new RandomGenerator.Long(), size)); 
print("a6 = ”+ Arrays.toString(a6)); 
float(] a7 = ConvertTo.primitive(Generated.array( 
Float.class, new RandomGenerator.Float(), size)); 
print("a7 = ”+ Arrays.toString(a7)); 
double[] a8 = ConvertTo. primitive (Generated. array( 
Double.class, new RandomGenerator.Double(), size)); 
print("a8 = ”+ Arrays. toString(a8)); 
} 
} /* Output: 
al = {true, false, true, false, false, true] 
a2 = (104, -79, -76, 126, 33, -64] 








了 
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a3 = [Z, n, T, c, Q. r] 

a4 = [-13408, 22612, 15401, 15161, -28466, -12683] 
aS = (7704, 7383, 7706, 575, 8410, 6342] 

a6 = [7674, 8804, 8950, 7826, 4322, 896] 

0.01, 0.2, @.4, 0.79, 0.27, 6.45) 

0.16, 0.87. 0.7, 0.66, 0.87, 0.59] 





FT, 

这 些 测试 还 可 以 确保 ConvertTo.primitive0 方 法 的 每 个 版 本 都 可 以 正确 地 工作 。 

练习 11: (2) 展示 自动 包装 机 制 不 能 应 用 于 数组 。 

练习 12: (1) 用 CountingGenerator 创 建 一 个 初始 化 过 的 double 数 组 并 打印 结果 。 

45513; (2) 用 CountingGenerator.Character 填 充 一 个 String。 

练习 14，(6) 对 每 个 基本 类 型 都 创建 一 个 数组 ， 然 后 用 CountingGenerator 来 填充 每 个 数组 
并 打印 所 有 的 数组 。 

练习 15，(2) 修改 ContainerComparison.java， 创 建 一 个 用 于 BerylliumSphere 的 Generator， 
并 修改 main0 方 法 ， 再 将 这 个 Generator 作 用 于 Generated.array0。 

练习 16，(3) 从 CountingGeneratorjava 开 始 ， 创 建 一 个 SkipGenerator 类 ， 它 可 以 根据 构造 
器 参数 ， 通 过 递增 产生 新 值 。 修 改 TestArrayGenerationjava， 以 展示 新 类 可 以 正确 地 工作 。 

练习 17: (5) 创建 并 测试 用 于 BigDecimal 的 Generator， 并 确保 它 可 以 用 于 Generated 中 的 
方法 。 


16.7 Arrays 实 用 功能 


在 java.util 类 库 中 可 以 找到 Arrays 类 ， 它 有 一 套用 于 数组 的 static 实 用 方法 ， 其 中 有 六 个 基 
本 方法 :equals0 用 于 比较 两 个 数组 是 否 相等 (deepEquals0 用 于 多 维 数组 ) ，fill0 在 本 章 前 面 
部 分 已 经 论述 过 了 ，sort0 用 于 对 数组 排序 ，binarySearch() 用 于 在 已 经 排序 的 数组 中 查找 元 
素 ，toString0 产 生 数组 的 String 表 示 ，hashCode0 产 生 数 组 的 散 列 码 (你 将 在 第 17 章 中 学 习 它 ) 。 
所 有 这 些 方法 对 各 种 基本 类 型 和 Object 类 而 重 载 过 。 此 外 ，Arrays.asList0 接 受 任意 的 序列 或 数 
组 作为 其 参数 ， 并 将 其 转变 为 List 容 器 一 这 个 方法 在 第 11 章 中 已 经 介绍 过 了 。 

在 讨论 Arrays 的 方法 之 前 ， 我 们 先 看 看 另 一 个 不 属于 Arrays 但 很 有 用 的 方法 。 
16.7.1 复制 数组 

Java 标 准 类 库 提 供 有 static 方 法 System.arraycopy0, 用 它 复制 数组 比 用 for 循 环 复 制 要 快 很 多 。 
System.arraycopy0 针 对 所 有 类 型 做 了 重 载 。 下 面 的 例子 就 是 用 来 处 理 int 数 组 的 ， 


1/: arrays/CopyingArrays.java 
// Using System.arraycopy() 





import java.util.*; 
import static net.mindview.util.Print.*; 


public class CopyingArrays { 
public static void main(String{] args) { 
int(] 1 = new int[7]; 
int[] j = new int[19]: 
Arrays. fill (i, 47): 
Arrays. fill(j, 99); 
print("i = " + Arrays. toString(i)); 





”+ Arrays. toString(j)): 
System.arraycopy(i, 0. j, 8, i.length); 
print("j = " + Arrays. toString(j)): 
int[] k = new int[5]; 

Arrays. fill(k, 163); 
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system.arraycopy(i, @. k, 8, k.length); 
print("k =" + Arrays. toString(k)): 

Arrays. fill(k, 183); 

System.arraycopy(k, 9, i, @, k.length); 
print("i = ”+ Arrays. toString(i)): 

7/ Objects: 

Integer(] u = new Integer [19]: 

Integer{] v = new Integer [5] ; 

Arrays.fill(u, new Integer(47)): 

Arrays.fill(v, new Integer (99))+ 

print("u = " + Arrays. toString(u)); 

print("v = " + Arrays. toString(v)): 
System.arraycopy(v, @, u, u.length/2, v. length); 
print("u = " + Arrays.toString(u)); 


} /* Output: 

i = (47, 47, 47, 47, 47, 47, 47) 

j = [99, 99, 99, 99, 99, 99, 99, 99, 99, 99] 
j = (47, 47, 47, 47, 47, 47, 47, 99, 99, 99] 
k = (47, 47, 47, 47, 47) 

4 = (103, 103, 103, 103, 103, 47, 47] 

u = (47, 47, 47, 47, 47, 47, 47, 47, 47, 47) 
v = (99, 99, 99, 99, 99] 

u = [47, 47, 47, 47, 47, 99, 99, 99, 99, 99) 
Whim 


arraycopy() 需 要 的 参数 有 : WBA, 表示 从 源 数 组 中 的 什么 位 置 开始 复制 的 偏 移 量 ， 表 示 
从 目标 数组 的 什么 位 置 开始 复制 的 偏 移 量 ， 以 及 需要 复制 的 元 素 个 数 。 当然 ， 对 数组 的 任何 越 
界 操作 都 会 导致 异常 。 

这 个 例子 说 明基 本 类 型 数组 与 对 象 数组 都 可 以 复制 。 然 而 ， 如 果 复 制 对 象 数组 ， 那 么 只 是 
复制 了 对 象 的 引用 一 而 不 是 对 象 本 身 的 拷贝 。 这 被 称 作 浅 复制 《shallow copy) (参见 本 书 的 在 
线 补充 材料 以 了 解 更 多 的 内 容 )。 

System.arraycopy0 不 会 执行 自动 包装 和 自动 拆 包 ， 两 个 数组 必须 具有 相同 的 确切 类 型 。 

练习 18，(3) 创建 并 填充 一 个 BerylliumSphere 数 组 ， 将 这 个 数组 复制 到 一 个 新 数组 中 ， 并 
展示 这 是 一 种 没 复制 。 

16.7.2 数组 的 比较 

Arrays 类 提供 了 重 载 后 的 equals0 方 法 ， 用 来 比较 整个 数组 。 同 样 ， 此 方法 针对 所 有 基本 类 
型 与 Object 都 做 了 重 载 。 数 组 相等 的 条 件 是 元 素 个 数 必 须 相 等 ， 并 且 对 应 位 置 的 元 素 也 相等 ， 
这 可 以 通过 对 每 一 个 元 素 使 用 equals0 作 比较 来 判断 。( 对 于 基本 类 型 ， 需要 使 用 基本 类 型 的 包 
装 器 类 的 equals0 方 法 ， 例 如 ， 对 于 int 类 型 使 用 Integer.equals0 作 比较 ) ILTA: 


11: arrays/ConparingArrays.java 
7/ Using Arrays.equals() 

import java.util.*; 

import static net.mindview.util.Print.*; 


public class ComparingArrays { 
public static void main(Stringl) args) { 

int{] al = new int(10}; 
int[] a2 = new int[19]; 
Arrays.fill(al, 47); 
Arrays. fill(a2, 47); 
print (Arrays.equals(al, a2)): 
a2(3] = 11; 
print(Arrays.equals(al, a2)): 
String{] s1 = new String(4]; 
Arrays. fill(st, "Hi"); 
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String{] s2 = { new String("Hi"), new String("Hi"), 
new String("Hi"), new String("Hi") }; 
print(Arrays.equals(si, s2)); 


} 
} /* Output: 
true 
false 
true 
Whim 


最 初 ，al 与 a2 完 全 相等 ， 所 以 输出 为 tue， 然 后 改变 其 中 一 个 元 素 ， 使 得 结果 为 false。 在 最 
后 一 个 例子 中 ，s1 的 所 有 元 素 都 指向 同一 个 对 象 ， 而 数组 s2 包 含 五 个 相互 独立 的 对 象 。 然 而 ， 
数组 相等 是 基于 内 容 的 (通过 Object.equals0 比 较 )， 所 以 结果 为 true。 

练习 19: (2) 创建 一 个 类 ， 它 有 一 个 用 构造 器 中 的 参数 初始 化 的 int 域 。 创 建 由 这 个 类 的 对 象 
构成 的 两 个 数组 ， 每 个 数组 都 使 用 了 相同 的 初始 化 值 ， 然 后 展示 它们 不 相等 的 Arrays.equals0) 声 
明 。 在 你 的 类 中 添加 一 个 equals0 方 法 来 解决 此 问题 。 

练习 20，(4) 演示 用 于 多 维 数组 的 deepEquals0 方 法 。 

16.7.3 数组 元 素 的 比较 

排序 必须 根据 对 象 的 实际 类 型 执行 比较 操作 。 一 种 自然 的 解决 方案 是 为 每 种 不 同 的 类 型 各 
编写 一 个 不 同 的 排序 方法 ， 但 是 这 样 的 代码 难以 被 新 的 类 型 所 复 用 。 

程序 设计 的 基本 目标 是 “将 保持 不 变 的 事物 与 会 发 生 改变 的 事物 相 分 离 "， 而 这 里 ， 不 变 的 
是 通用 的 排序 算法 ， 变 化 的 是 各 种 对 象 相互 比较 的 方式 。 因 此 ， 不 是 将 进行 比较 的 代码 编写 成 
不 同 的 子 程序 ， 而 是 使 用 策略 设计 模式 。 通 过 使 用 策略 ， 可 以 将 “会 发 生变 化 的 代码 ”封装 
在 单独 的 类 中 (策略 对 象 )， 你 可 以 将 策略 对 象 传递 给 总 是 相同 的 代码 ， 这 些 代码 将 使 用 策略 来 
完成 其 算法 。 通 过 这 种 方式 ， 你 能 够 用 不 同 的 对 象 来 表示 不 同 的 比较 方式 ， 然 后 将 它们 传递 给 
相同 的 排序 代码 。 

Java 有 两 种 方式 来 提供 比较 功能 。 第 一 种 是 实现 javalang.Comparable 接 口 ， 使 你 的 类 具有 
“天 生 ” 的 比较 能 力 。 此 接口 很 简单 ， 只 有 compareTo0 一 个 方法 。 此 方法 接收 另 一 个 Object 为 
参数 ， 如 果 当 前 对 象 小 于 参数 则 返回 负 值 ， 如 果 相等 则 返回 零 ， 如 果 当 前 对 象 大 于 参数 则 返回 
正 值 。 

下 面 的 类 实现 了 Comparable 接 口 ， 并 且 使 用 Java 标 准 类 库 的 方法 Arrays.sort0 演 示 了 比较 的 
BR: 


//: arrays/CompType. java 
// Implementing Comparable in a class. 
import java.util.*; 

import net.mindview.util.*; 

import static net.mindview.util.Print.*; 


public class CompType implements Comparable<CompType> { 
int i; 
int j: 
private static int count = 1; 
public CompType(int nl, int n2) { 
i = nl; 
j = n2; 
} 
public String toString() { 
String result = "i =" +i +", fete jest; 
if(count++ % 3 == @) 


© 参考 Erich Gammafty (Design Patiem》 和 wwwMindView 上 的 《Thinking in Pattern (With Java)), J} (Design 
Pattern) 的 中 文 版 、 英 文 影印 版 及 双语 线 均 已 由 机 械 工业 出 版 社 出 版 一 编辑 注 。 
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result += "\n"; 
return result; 


} 
public int compareTo(CompType rv) { 
return (i < rv.i ? -1 : (i == rv.i?0: 1); 
} 
private static Random r = new Random(47); 
public static Generator<CompType> generator () { 
return new Generator<CompType>() { 
public CompType next() { 
return new CompType(r.nextInt(100),r.nextInt(100)); 
} 
和 
} 
public static void main(String[] args) { 
CompTypel] a = 
Generated. array (new CompType{12], generator ()): 
print("before sorting:"); 
print(Arrays.toString(a)); 
Arrays.sort(a) ; 
print("after sorting:"): 
print(Arrays. toString(a)); 


} 
} /* Output: 
before sorting: 
LI = 58, j = 55), (i = 93, j = 61], [i = 61, j = 29) 
. = 68, f= 8), [i = 22, j= 7), [i = 88, j = 28) 
» [i = 51, j = 89], [i = 9, j = 78], [i = 98, j = 61) 
, [i = 26, j = 58), [i = 16, j = 40), [i = 11, j = 22) 


after sorting: 

[H = 9, j = 78), (1 = 11, j = 222, (i = 16, j = 40] 

. U2 20, j = 58), [i = 22, j = 7], [i = Si, j = 89) 

. H = 58, j = 55], [1 = 61, j = 291, [i = 68, j = 0) 

, [i = 88, j = 28), [i = 93, j = 61), [i = 98, j = 61) 

Yr- A 

在 定义 作 比较 的 方法 时 ， 由 你 来 负责 决定 将 你 的 一 个 对 象 与 另 一 个 对 象 作 比 较 的 含义 。 这 
里 在 比较 中 只 用 到 了 i 值 ， 而 忽略 了 j 值 。 

generator() 方 法 生成 一 个 对 象 ， 此 对 象 通过 创建 一 个 匿名 内 部 类 ( 见 第 8 章 ) 来 实现 
Generator 接 口 。 该 例 中 构建 CompType 对 象 ， 并 使 用 随机 数 加 以 初始 化 。 在 main0 中 ， 使 用 生 
成 器 填充 CompType 的 数组 ， 然 后 对 其 排序 。 如 果 没 有 实现 Comparable 接 口 ， 调 用 sort0 的 时 候 
会 抛 出 ClassCastException 这 个 运行 时 异常 。 因 为 sort0 需 要 把 参数 的 类 型 转变 为 Comparable。 

假设 有 人 给 你 一 个 并 没有 实现 Comparable 的 类 ， 或 者 给 你 的 类 实现 了 Comparable， 但 是 你 
不 喜欢 它 的 实现 方式 ， 你 需要 另外 一 种 不 同 的 比较 方法 。 要 解决 这 个 问题 ， 可 以 创建 一 个 实现 
了 Comparator 接 口 〈 在 第 11 章 中 简要 介绍 过 ) 的 单独 的 类 。 这 是 策略 设计 模式 的 一 个 应 用 实例 。 
这 个 类 有 compare0 和 equals0 两 个 方法 。 然 而 ， 不 一 定 要 实现 equals0 方 法 ， 除 非 有 特殊 的 性 能 
需要 ， 因 为 无 论 何 时 创建 一 个 类 ， 都 是 间接 继承 自 Object， 而 Object 带 有 equals( 方 法 。 所 以 只 
需 用 默认 的 Object 的 equals0 方 法 就 可 以 满足 接口 的 要 求 了 。 

Collections 类 (在 下 一 章 会 详细 介绍 ) 包含 一 个 reverseOrder0 方 法 ， 该 方法 可 以 产生 一 个 
Comparator， 它 可 以 反 转自 然 的 排序 顺序 。 这 很 容易 应 用 于 CompType: 


//: arrays/Reverse.java 
// The Collections.reverseOrder() Comparator 
import java.util.*: 
import net.mindview.util.*; 

import static net.mindview.util.Print.*; 








public class Reverse { 
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public static void main(String[] args) { 

CompType[] a = Generated. array ( 
new CompType[12). CompType. generator 0) : 

print("before sorting:"); 
print(Arrays.toString(a)) 
Arrays.sort(a, Collections. reverseOrder()); 
print("after sorting:"); 
print (Arrays. toString(a)) ; 





} 
} /* Output: 
before sorting: 


(fi = 58, j = 55], [i = 93, j = 61), 29) 
. [i = 68, j = 0}. [i = 22, j= 71. y 28) 

, B = 51, j = 89), [i = 9, j = 78), [i = 98, j = 61) 
, [1 = 20, j = 58), [i = 16, j = 40], [i = 12, j = 22] 


] 
after sorting: 





[i = 98, j = 61), [i = 93, j = 61], = 28) 
» UW = 68, j = 6}, [i = 61, j = 29), = 55] 
» [1 = 51, j = 89), = 58] 
, [t= 16, j = 40), [i = 11, j = 78) 
1 

Whim 


也 可 以 编写 自己 的 Comparator。 在 这 里 的 CompType 对 象 是 基于 j 值 而 不 是 基于 i 值 的 。 


/1/: arrays/ComparatorTest. java 
// Implementing a Comparator for a class. 
import java.util. *; 

import net.mindview.util. 
import static net.mindview. 





til.Print.*; 





class CompTypeComparator implements Comparator<CompType> { 
public int compare(CompType 01, CompType 02) { 

i return (ol.j < 02.j ? -1 : (01.j == 02.j ? 8 : 1)); 

} 





A ) 


public class ComparatorTest { 
public static void main(String(} args) { 
CompType[] a = Generated. array( 
new CompType(12]. CompType.generator()); 

print("before sorting:"); 
print (Arrays. toString(a)); 
Arrays.sort(a, new CompTypeComparator()); 
print("after sorting:"); 
print(Arrays.toString(a)); 


} 
} /* Output: 
before sorting: 





[i = 58, j = 55), [i = 93, j j = 29) 
, (i= 68,1=0], [Í = 22. j A . j = 28) 
， [i = 51, j = 89). [1 =9, j é 98, j = 61) 
, (i = 20, j = 58). [i = 16, j ,i= 11. j = 227 


] 

after sorting: 

[ti = 68, j = 9]，[i = 
. Ui = 88, j = 28), [i 
. D = 58, j = 55), [i 
. [i = 98, j = 61), fi 


nun 





1 
Whim 
练习 21: (3) 试 着 对 练习 18 中 的 对 象 数组 进行 排序 。 
16.7.4 数组 排序 
使 用 内 置 的 排序 方法 ， 就 可 以 对 任意 的 基本 类 型 数组 排序 ， 也 可 以 对 任意 的 对 象 数组 进行 
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排序 ， 只 要 该 对 象 实现 了 Comparable 接 口 或 具有 相关 联 的 Comparators 。 下 面 的 例子 生成 随机 
的 String 对 象 ， 并 对 其 排序 : 


//: arrays/StringSorting. java 
// Sorting an array of Strings. 

import java.util.*; 

import net.mindview.util.*; 

import static net.mindview.util.Print.*: 


public class StringSorting { 
public static void main(String[] args) { 
String{] sa = Generated. array(new String(26), 
new RandomGenerator.String(5)); 

print("Before sort: " + Arrays.toString(sa)); 
Arrays.sort(sa); 
print("After sort: " + Arrays.toString(sa)); 
Arrays.sort(sa, Collections. reverseOrder()); 
print("Reverse sort: " + Arrays. toString(sa)): 
Arrays.sort(sa, String.CASE_INSENSITIVE_ORDER) ; 
print ("Case-insensitive sort: ”+ Arrays. toString(sa)); 


} 
} /* Output: 
Before sort: [YNzbr, nyGcF, OWZnT, cQrGs, eGZMm, JMRoE, 
SuECU, OneOE, dismw, HLGEa, hKcxr, EqUCB, bkIna, Mesbt, 
WHKjU, rUKZP, gwsqP, zDyCy, RFJQA, HxxHv] 
After sort: [EqUCB, HLGEa, HxxHv, JMROE, Mesbt, OWZnT, 
OneOE, RFJQA, WHkjU, YNzbr, bkIna, cQrGs, dLsmw, eGZMm, 
gwsqP, hKcxr, nyGcF, rUkZP, suEcU, zDyCy] 
Reverse sort: [zDyCy, suEcU, rUkZP, nyGcF, hKcxr, gwsqP. 
eGZMm, dlsmw, cQrGs, bkIna, YNzbr, WHkjU, RFJQA, OneOE, 
OWZnT, Mesbt, JMROE, HxxHv, HLGEa, EqUCB) 
Case-insensitive sort: [bkIna, cQrGs, dLsmw, eGZMm, EqUCB, 
gwsqP, hKexr, HLGEa, HxxHv, JMROE, Mesbt, nyGcF, OneOE, 
OWZnT, RFJQA, rUKZP, suEcU, WHkjU, YNzbr, zDyCy) 
Whim 


注意 ，String 排 序 算法 依据 词典 编排 顺序 排序 ， 所 以 大 写字 母 开 头 的 词 都 放 在 前 面 输出 ， 然 
后 才 是 小 写字 母 开头 的 词 。( 电 话 舌 通常 是 这 样 排序 的 。) 如 果 想 忽略 大 小 写字 母 将 单词 都 放 在 
一 起 排序 ， 那 么 可 以 像 上 例 中 最 后 一 个 对 sort0 的 调用 一 样 ， 使 用 String。CASE_INSENSITIVE_ 
ORDER, 

Java 标 准 类 库 中 的 排序 算法 针对 正 排序 的 特殊 类 型 进行 了 优化 一 针对 基本 类 型 设计 的 “ 快 
速 排序 ”(Quicksort) ， 以 及 针对 对 象 设计 的 “稳定 归并 排序 "。 所 以 无 须 担心 排序 的 性 能 ， 除 非 
你 可 以 证 明 排 序 部 分 的 确 是 程序 效率 的 瓶颈 。 

16.7.5 在 已 排序 的 数组 中 查找 

如 果 数 组 已 经 排 好 序 了 ， 就 可 以 使 用 Arrays.binarySearch0 执 行 快速 查找 。 如 果 要 对 未 排序 
的 数组 使 用 binarySearch()， 那 么 将 产生 不 可 预料 的 结果 。 下 面 的 例子 使 用 RandIntGenerator. 
Integer 填 充 数组 ， 然 后 再 使 用 同样 的 生成 器 生成 要 查找 的 值 : 


1/1: arrays/ArraySearching. java 
// Using Arrays.binarySearch(). 

‘import java.util.*; 

import net.mindview.util.*; 

‘import static net.mindview.util.Print.*; 


public class ArraySearching { 
Public static void main(String{] args) { 


日 令 人 惊奇 的 是 ，Java 1.0 或 1.1 对 String 排 序 未 提供 任何 支持 。 
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Generator<Integer> gen = 
new RandomGenerator . Integer (1000) ; 
int[] a = ConvertTo.primitive( 
Generated.array(new Integer [25], gen)); 
Arrays. sort (a); 
print("Sorted array: " + Arrays.toString(a)); 
while(true) { 
int r = gen.next(); 
int location = Arrays.binarySearch(a, r); 
if(location >= 0) { 
print("Location of "+r +" is * + location + 
", al" + location + "] = * + allocation]); 
break; // Out of while loop 
} 
上 
} 
} /* Output: 
Sorted array: [128, 140, 26, 207, 258, 258, 278, 288, 322. 
429, 511, 526, 522, 551, 555, 589, 693, 704, 809, 861, 861, 
868, 916, 961, 998) 
Location of 322 is 8, a[8] = 322 
Whim 


在 while 循 环 中 随机 生成 一 些 值 作 为 查找 的 对 象 ， 直 到 找到 一 个 才 停 止 循环 。 

如 果 找到 了 目标 ，Arrays.binarySearch0 产 生 的 返回 值 等 于 或 大 于 0。 否 则 ， 它 产生 负 返 回 
表示 若 要 保持 数组 的 排序 状态 此 目标 元 素 所 应 该 插入 的 位 置 。 这 个 负 值 的 计算 方式 是 ; 

- (插入 点 ) -1 

“插入 点 ”是 指 ， 第 一 个 大 于 查找 对 象 的 元 素 在 数组 中 的 位 置 ， 如 果 数 组 中 所 有 的 元 素 都 小 
于 要 查找 的 对 象 ,“ 插 人 点 ”就 等 于 asize0 。 

如 果 数组 包含 重复 的 元 素 ， 则 无 法 保证 找到 的 是 这 些 副本 中 的 哪 一 个 。 搜 索 算 法 确实 不 是 
专 为 包含 重复 元 素 的 数组 而 设计 的 ， 不 过 仍然 可 用 。 如 果 需 要 对 没有 重复 元 素 的 数组 排序 ， 可 
以 使 用 TreeSet (保持 排序 顺序 ) ， 或 者 LinkedHashSet (保持 插入 顺序 )， 后 面 我 们 将 会 介绍 它 
们 。 这 些 类 会 自动 处 理 所 有 的 细节 。 除 非 它们 成 为 程序 性 能 的 瓶颈 ， 否 则 不 需要 自己 维护 数组 。 

如 果 使 用 Comparator 排 序 了 某 个 对 象 数组 (基本 类 型 数组 无 法 使 用 Comparator 进 行 排序 ) , 
在 使 用 binarySearchO 时 必须 提供 同样 的 Comparator (使 用 binarySearch (方法 的 重 载 版 本 ) 。 
例如 ， 可 以 修改 StringSorting.java 程 序 以 进行 某 种 查找 : 


11: arrays/AlphabeticSearch.java 
// Searching with a Comparator. 
import java.util.*; 

import net.mindview.util.*; 


值 


public class AlphabeticSearch { 
public static void main(Stringl] args) { 

String[] sa = Generated.array(new String[36] , 
new RandonGenerator .String(5)); 

Arrays.sort(sa, String.CASE_INSENSITIVE_ORDER) ; 

System.out.printin (Arrays. toString(sa)); 

int index = Arrays.binarySearch(sa, sa(10], 
String.CASE_INSENSITIVE_ORDER) ; 

System.out.printin("Index: "+ index + "\n"+ salindex]); 


} 
} /* Output: 
[bkIña, cQrGs, cXZJo, dlsmw, eGZMm, EqUCB. gwsqP, hKcxr, 
HLGEa, HqXum, HxxHv, JMRoE, JmzMs, Mesbt, MNvqe, nyGcF, 
OgoYW, OneOE, OWZnT, RFJQA, rUKZP, sgqia, slJrl, suEcU, 
uTpnX, vpfFv. WHkjU, xxEAJ, YNzbr, zDyCy] 
Index: 10 
HxxHy 
Ii~ 
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这 里 的 Comparator 必 须 接受 重 载 过 的 binarySearch0 作 为 其 第 三 个 参数 。 在 这 个 例子 中 ,由 
于 要 查找 的 目标 就 是 从 数组 中 选 出 来 的 元 素 ， 所 以 肯定 能 查找 到 。 

练习 22: (D 通过 程序 说 明 在 未 排序 数组 上 执行 binarySearch( 方 法 的 结果 是 不 可 预知 的 。 

练习 23: (2) 创建 一 个 Integer 数 组 ， 用 随机 的 int 数 值 填 充 它 (使 用 自动 包装 机 制 )， 再 使 用 
Comparator 将 其 进行 反 向 排序 。 

练习 24: (3) 通过 程序 说 明 练 习 19 中 的 类 是 可 查找 的 。 


16.8 总 结 


在 本 章 中 ， 你 看 到 了 Java 对 尺寸 固定 的 低级 数组 提供 了 适度 的 支持 。 这 种 数组 强调 的 是 性 
能 而 不 是 灵活 性 ， 并 且 与 C 和 C++ 的 数组 模型 类 似 。 在 Java 的 初始 版 本 中 ， 尺 寸 固定 的 低级 数组 
绝对 是 必需 的 ， 不 仅 是 因为 Java 的 设计 者 选择 在 Java 中 要 包含 基本 类 型 (也 是 出 于 性 能 方面 的 考 
处)， 而 且 还 因为 那个 版 本 中 对 容器 的 支持 非常 少 。 因 此 ， 在 Java 的 早期 版 本 中 ， 选 择 包含 数组 
总 是 合理 的 。 

其 后 的 Java 版 本 对 容器 的 支持 得 到 了 明显 的 改进 ， 并 且 现 在 的 容器 在 除了 性 能 之 外 的 各 个 
方面 都 使 得 数组 相形 见 纳 。 就 像 在 本 书 其 他 多 处 地 方 所 叙述 的 那样 ， 对 你 来 说 ， 性 能 出 问题 的 
地 方 通常 是 无 论 如 何 你 都 无 法 想象 得 到 的 。 

有 了 额外 的 自动 包装 机 制 和 放 型 ， 在 容器 中 持 有 基本 类 型 就 变 得 易如反掌 了 ， 而 这 也 进 一 
步 促 使 你 用 容器 来 替换 数组 。 因 为 泛 型 可 以 产生 类 型 安全 的 容器 ， 因 此 数组 面 对 这 一 变化 ， 已 
经 变 得 毫 无 优势 了 。 

就 像 在 本 章 中 描述 的 ， 而 且 当 你 尝试 着 使 用 它们 时 也 会 看 到 ， 泛 型 对 数组 是 极 大 的 威胁 。 
通常 ， 即 使 当 你 可 以 让 泛 型 与 数组 以 某 种 方式 一 起 工作 时 (在 下 一 章 你 将 会 看 到 )， 在 编译 期 你 
最 终 也 会 得 到 “不 受 检查 ”的 警告 信息 。 

曾经 在 多 个 场合 ， 当 我 和 Java 语 言 的 设计 者 们 讨论 某 些 特定 的 示例 时 ， 我 直接 告诉 他 们 ， 
我 应 该 使 用 容器 而 不 是 数组 〈 在 这 些 示例 中 ， 我 使 用 数组 是 为 了 演示 某 些 具体 的 技术 ， 因 此 我 
没有 选择 的 余地 ) 。 

所 有 这 些 话题 都 表示 : 当 你 使 用 最 近 的 Java 版 本 编程 时 ， 应 该 “优选 容器 而 不 是 数组 "。 只 
有 在 已 证 明 性 能 成 为 问题 (并且 切 换 到 数组 对 性 能 提高 有 所 帮助 ) 时 ， 你 才 应 该 将 程序 重 构 为 
使 用 数组 。 

这 是 一 个 相当 清晰 的 陈述 ， 但 是 有 些 语言 根本 就 没有 尺寸 固定 的 低级 数组 ， 它 们 只 有 尺寸 
可 调 的 容器 ， 这 些 容器 与 C/C++/Java 风 格 的 数组 相 比 ， 明 显 具 有 更 多 的 功能 。 例 如 ，Pythone 具 
有 一 个 使 用 基本 数组 语法 的 List 类 型 ， 但 是 它 具 有 更 多 功能 一 -你 甚至 可 以 继承 它 。 


#: arrays/PythonLists.py 





alist = [1, 2, 3, 4, 5] 

print type(aList) # <type 'list'> 

print alist # [1, 2, 3, 4, 5] 

print alist[4] #5 Basic list indexing 
alist.append(6) # lists can be resized 
alist += [7, 8] # Add a list to a list 
print alist # (1, 2, 3, 4, 5. 6, 7, 8] 
aSlice = alist(2:4] 

print aSlice # [3, 4] 


class MyList(list): # Inherit from list 


© 查看 wwwPython.org。 
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# Define a method, ‘this’ pointer is explicit: 
def getReversed(self): 
reversed = self{:] # Copy list using slices 
reversed.reverse() # Built-in list method 
return reversed 


list2 = MyList(alist) # No ‘new’ needed for object creation 

print type(list2) # <class ‘_main_.MyList'> 

print list2.getReversed() # (8, 7, 6, 5, 4, 3, 2, 1] 

前 一 章 已 经 介绍 过 基本 的 Python 语法 。 在 本 例 中 ， 直 接 通过 用 方 括号 括 起 来 的 且 由 逗号 分 
割 的 对 象 序列 ， 创 建 了 一 个 列表 ， 所 产生 的 结果 就 是 运行 时 类 型 为 List 的 一 个 对 象 (print 语 名 
的 输出 如 同一 行 的 注释 所 示 )。 打 印 List 的 结果 与 使 用 Java 中 的 Arrays.toString0 相 同 。 

创建 List 的 子 序列 是 通过 在 索引 操作 的 内 部 放置 “:” 操 作 符 ， 从 而 用 “切片 ”来 实现 的 。 
List 类 型 具有 很 多 内 置 的 操作 。 

MyList 是 一 个 类 定义 ， 在 括号 内 的 是 其 基 类 。 在 这 个 类 的 内 部 ，def 语句 将 产生 方法 ， 而 方 
法 的 第 一 个 参数 自动 地 与 Java 中 的 this 等 价 ， 只 是 在 Python 中 它 是 显 式 的 ， 并 且 按 惯例 其 标识 符 
为 self (这 不 是 关键 字 )。 请 注意 构造 器 是 如 何 自动 继承 的 。 

尽管 Python 中 的 每 项 事物 确实 都 是 对 象 (包括 整 型 和 浮 点 类 型 ) ， 但 是 仍旧 有 其 他 出 口 ， 使 
得 你 可 以 去 优化 代码 中 性 能 关键 的 部 分 ， 这 时 需要 用 C 或 C++ 编写 一 些 扩展 ， 或 者 使 用 被 称 为 
Pyrex 的 特殊 工具 ， 它 被 设计 用 来 让 我 们 更 方便 地 提高 代码 的 执行 速度 。 通 过 这 种 方式 ， 你 仍旧 
可 以 保持 对 象 的 纯粹 性 ， 同 时 又 不 妨碍 对 性 能 的 改进 。 

PHP 语 言 * 走 得 更 远 ， 它 只 有 单一 的 数组 类 型 ， 即 可 以 充当 用 int 来 索引 的 数组 ， 也 可 以 充 
当 关 联 数组 (Map)。 

在 Java 不 断 演化 了 许多 年 之 后 ， 研 究 这 样 一 个 问题 会 相当 有 趣 ， 如 果 Java 的 设计 者 们 能 够 从 
头 再 来 ， 他 们 是 否 还 会 在 Java 语 言 中 设计 基本 类 型 的 低级 数组 。 如 果 当 初 抛弃 它们 ，Java 也 许 就 
会 成 为 真正 的 纯 面向 对 象 语言 (不 管 人 们 如 何 宣传 ，Java 仍 旧 不 是 纯 面向 对 象 语言 ， 而 原因 正 
是 这 些 低级 的 绊脚石 )。 最 初 有 关 效 率 的 论点 总 是 很 吸引 人 ， 但 是 随 着 时 间 的 推移 ， 我 们 看 到 了 
与 这 种 思想 背道而驰 ， 向 着 使 用 像 容 器 这 类 高 级 构件 的 方向 的 演化 。 另 外 ， 如 果 容器 能 够 像 某 
些 语言 一 样 内 置 于 语言 的 内 核 中 ， 那 么 编译 器 就 会 得 到 更 好 的 优化 良机 。 

我 们 肯定 还 会 使 用 数组 ， 并 且 你 在 读 代码 的 时 全 还 会 看 到 它 ， 但 是 ， 容 器 几乎 总 是 更 好 的 
选择 。 

练习 25，(3) 用 Java 重 写 PythonLists.py。 

所 选 习 题 的 答案 都 可 以 在 名 为 The Thinking in Java Annotated Solution Guide 的 电子 文档 中 找 
到 ， 读 者 可 以 从 www.MindView.net 处 购买 此 文档 。 


日 查看 www.php.net。 





第 17 章 容器 深入 研究 

第 11 章 介绍 了 Java 容 器 类 库 的 概念 和 基本 功能 ， 这 些 对 于 使 用 容器 来 说 已 经 足够 了 。 本 章 
将 更 深入 地 探索 这 个 重要 的 类 库 。 

为 了 充分 利用 容器 类 库 ， 你 需要 了 解 比 第 11 章 中 介绍 的 内 容 更 多 的 知识 ， 但 是 本 章 依赖 于 
高 级 特性 (例如 泛 型 )， 因 此 被 安排 在 了 全 书 较为 靠 后 的 位 置 。 

在 对 容器 有 了 更 加 完备 的 了 解 之 后 ， 你 将 学 习 散 列 机 制 是 如 何 工作 的 ， 以 及 在 使 用 散 列 容 
器 时 怎样 编写 hashCode0 和 equals( 方 法 。 你 还 将 学 习 为 什么 某 些 容器 会 有 不 同 版 本 的 实现 ， 以 
及 怎样 在 它们 之 间 进 行 选 择 。 本 章 最 后 将 以 对 通用 便利 工具 和 特殊 类 的 探索 作为 结束 。 


17.1 完整 的 容器 分 类 法 


第 11.14 节 展示 了 Java 容 器 类 库 的 简化 图 。 下 面 是 集合 类 库 更 加 完备 的 图 。 包 括 抽象 类 和 遗 
留 构 件 (不 包括 Queue 的 实现 ): 
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Full Container Taxonomy 


Java SE5 新 添加 了 : 

* Queue 接 口 (正如 在 第 11 章 所 介绍 ，LinkedList 已 经 为 实现 该 接口 做 了 修改 ) 及 其 实现 
PriorityQueue 和 各 种 风格 的 BlockingQueue， 其 中 BlockingQueue 将 在 第 21 章 中 介绍 。 
"ConcurrentMap 接 口 及 其 实现 ConcurrentHashMap， 它 们 也 是 用 于 多 线程 机 制 的 ， 同 样 
会 在 第 21 章 中 介绍 。 
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"CopyOnWriteArrayList 和 CopyOn WriteArraySet， 它 们 也 是 用 于 多 线程 机 制 的 。 

* EnumSet 和 EnumMap， 为 使 用 enum 而 设计 的 Set 和 Map 的 特殊 实现 ， 将 在 第 19 章 中 介绍 。 

* 在 Collections 类 中 的 多 个 便利 方法 。 

虚线 框 表示 abstract 类 ， 你 可 以 看 到 大 量 的 类 的 名 字 都 是 以 Abstract 开 头 的 。 这 些 类 可 能 初 
看 起 来 有 点 令 人 困惑 ， 但 是 它们 只 是 部 分 实现 了 特定 接口 的 工具 。 例 如 ， 如 果 你 在 创建 自己 的 
Set， 那 么 并 不 用 从 Set 接 口 开 始 并 实现 其 中 的 全 部 方法 ， 只 需 从 AbstractSet 继 承 ， 然 后 执行 一 些 
创建 新 类 必需 的 工作 。 但 是 ， 事 实 上 容器 类 库 包含 足够 多 的 功能 ， 任 何 时 刻 都 可 以 满足 你 的 需 
求 ， 因 此 你 通常 可 以 忽略 以 Abstract 开 头 的 这 些 类 ， 


17.2 填充 容器 


虽然 容器 打印 的 问题 解决 了 ， 容 器 的 填充 仍然 像 java.util.Arrays 一 样 面临 同样 的 不 足 。 就 
像 Arrays 一 样 ， 相 应 的 Collections 类 也 有 一 些 实用 的 static 方 法 ， 其 中 包括 flO。 与 Arrays 版 本 
一 样 ， 此 fill( 方 法 也 是 只 复制 同一 个 对 象 引用 来 填充 整个 容器 的 ， 并 且 只 对 List 对 象 有 用 ， 但 是 
所 产生 的 列表 可 以 传递 给 构造 器 或 addAll0 方 法 : 


//:_containers/FillingLists. java 
// The Collections.fill() & Collections.nCopies() methods, 
import java.util.*; 


class StringAddress { 
private String s; 
public StringAddress(String s) { this.s = s; } 
public String toString() { 
return super.toString() + " " +s; 
} 
} 


public class FillingLists { 
public static void main(String[] args) { 
List<StringAddress> list= new ArrayList<StringAddress>( 
Collections.nCopies(4, new StringAddress("Hello"))); 
System.out.printin(List); 
Collections. fill(list, new StringAddress(*World!")); 
System.out.printin(list); 
} 
} /* Output: (Sample) 
[StringAddress@82ba41 Hello, StringAddress@82ba41 Hello, 
StringAddress@82ba41 Hello, StringAddress@82ba41 Hello] 
[StringAddress@923e30 World!, StringAddress@923e38 World!, 
StringAddress@923e30 World!, StringAddress@923e38 World!} 
Whim 


这 个 示例 展示 了 两 种 用 对 单个 对 象 的 引用 来 填充 Collection 的 方式 ， 第 一 种 是 使 用 
Collections.nCopies0 创 建 传递 给 构造 器 的 List， 这 里 填充 的 是 ArrayList。 

StringAddress 的 toString0 方 法 调用 Object.toString0 并 产生 该 类 的 名 字 ， 后 面 紧 跟 该 对 象 的 
散 列 码 的 无 符号 十 六 进 制 表示 (通过 hashCode0 生 成 的 )。 从 输出 中 你 可 以 看 到 所 有 引用 都 被 设 
置 为 指向 相同 的 对 象 ， 在 第 二 种 方法 的 Collection-fii0 被 调用 之 后 也 是 如 此 。 凶 10 方 法 的 用 处 更 
有 限 ， 因 为 它 只 能 替换 已 经 在 List 中 存在 的 元 素 ， 而 不 能 添加 新 的 元 素 。 
17.2.1 一 种 Generator 解 决 方案 

事实 上 ， 所 有 的 Collection 子 类 型 都 有 一 个 接收 另 一 个 Collection 对 象 的 构造 器 ， 用 所 接收 的 
Collection 对 象 中 的 元 素来 填充 新 的 容器 。 为 了 更 加 容易 地 创建 测试 数据 ， 我 们 需要 做 的 是 构建 
接收 Generator (在 第 15 章 中 定义 并 在 第 16 章 中 深入 探讨 过 ) 和 quantity 数 值 并 将 它们 作为 构造 
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器 参数 的 类 : 


/1: net/mindview/util/CollectionData. java 

// A Collection filled with data using a generator object. 
package net.mindview.util: 

import java.util.*; 


public class CollectionData<T> extends ArrayList<T> { 
public CollectionData(Generator<T> gen, int quantity) { 
for(int i = @; i < quantity; i++) 
add (gen.next()); 


// A generic convenience method: 
public static <T> CollectionData<T> 
List (Generator<T> gen, int quantity) { 
return new CollectionData<T>(gen, quantity); 





} 
yh 


这 个 类 使 用 Generator 在 容器 中 放置 所 需 数量 的 对 象 ， 然 后 所 产生 的 容器 可 以 传递 给 任何 
Collection 的 构造 器 ， 这 个 构造 器 会 把 其 中 的 数据 复制 到 自身 中 。addAll0 方 法 是 所 有 Collection 
子 类 型 的 一 部 分 ， 它 也 可 以 用 来 组 装 现 有 的 Collection。 

泛 型 便利 方法 可 以 减少 在 使 用 类 时 所 必需 的 类 型 检查 数量 。 

CollectionData 是 适配器 设计 模式 。 的 一 个 实例 ， 它 将 Generator 适 配 到 Collection 的 构造 
aL. 

下 面 是 初始 化 LinkedHashSet 的 一 个 示例 : 


//: containers/CollectionDataTest. java 
import java.util.*; 
import net.mindview.util.*; 


class Government implements Generator<String> { 
String(] foundation = ("strange women lying in ponds " + 
"distributing swords is no basis for a system of ”+ 
"government").split(" "); 
private int index; 
public String next() { return foundation{index++]; } 
} 


public class CollectionDataTest { 
public static void main(String[] args) { 
Set<String> set = new LinkedHashSet<String>( 
new CollectionData<String>(new Government(), 15)); 
// Using the convenience method: 
set.addAll (CollectionData.list(new Government(), 15)); 
System.out.printin(set); 











} 
} /* Output: 795 


[strange, women, lying, in, ponds, distributing, swords, 
is, no, basis, for, a, system, of, government} 
Whim 


这 些 元 素 的 顺序 与 它们 的 插入 顺序 相同 ， 因 为 LinkedHashSet 维 护 的 是 保持 了 插入 顺序 的 链接 
列表 。 


在 第 16 章 中 定义 的 所 有 操作 符 现在 通过 CollectionData 适 配器 都 是 可 用 的 。 下 面 是 使 用 了 其 
中 两 个 操作 符 的 示例 : 





O 与 《设计 模式 》 这 本 书 中 所 定义 的 适配器 相 比 ， 这 也 许 并 非 是 适配器 的 严格 定义 ， 但 是 我 认为 它 符合 适配器 
思想 的 基本 和 精神。 该 书 中 文 版 、 英 文 影印 版 与 双语 版 均 已 由 机 械 工业 出 版 社 出 版 。 一 一 编辑 注 
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//: containers/CollectionDataGeneration. java 

// Using the Generators defined in the Arrays chapter. 
import java.util. 
import net.mindview.util.*; 





public class CollectionDataGeneration { 
public static void main(String(] args) { 
System.out.printIn(new ArrayList<String>( 
CollectionData.list( // Convenience method 
new RandomGenerator.String(9), 19))); 
System. out.printin(new HashSet<Integer>( 
new CollectionData<Integer>( 
new RandomGenerator.Integer(), 18))); 
È 
} /* Output: 
[YNzbrnyGc, FOWZnTcQr, GseGZMmJM, RoEsuEcUO, neOEdLsmw, 
HLGEahKcx, rEqUCBbkI, naMesbtWH, kjUrUkZPg, wsqPzDyCy] 
[573, 4779, 871, 4367, 6090, 7882, 2017, 8637, 3455, 299] $ 
“i~ 


RandomGenerator.String 所 产生 的 String 长 度 是 通过 构造 器 参数 控制 的 。 
17.2.2 Map 生 成 器 

我 们 可 以 对 Map 使 用 相同 的 方式 ， 但 是 这 需要 有 一 个 Pair 类 ， 因 为 为 了 组 装 Map， 每 次 调 
用 Generator 的 next0 方 法 都 必须 产生 一 个 对 象 对 (一 个 键 和 一 个 值 ) ; 


//: net/mindview/util/Pair. java 
package net.mindview.util; 


public class Pair<K,V> { 
public final K key; 
public final V value; 
public Pair(K k, Vv) { 
key = k; 
value = v; 
} 
} ii~ 


key 和 value 域 都 是 public 和 final 的 ， 这 是 为 了 使 Pair 成 为 只 读 的 数据 传输 对 象 或 信使 ) 。 
Map 适 配器 现在 可 以 使 用 各 种 不 同 的 Generator、Iterator 和 常量 值 的 组 合 来 填充 Map 初 始 
化 对 象 : 


//: net/mindview/util/MapData. java 

// A Map filled with data using a generator object. 
package net.mindview.util; 

import java.util.*; 


public class MapData<K,V> extends LinkedHashMap<K.V> { 
/1/ A Single Pair Generator: 
public MapData(Generator<Pair<K.V>> gen, int quantity) { 
for(int i = 9; i < quantity; i++) { 
Pair<K,V> p = gen.next(); 
put(p.key; p.value); 
} 
$ 


// Two separate Generators: 
public MapData(Generator<K> genK, Generator<V> genV, 
int quantity) { 
for(int i =@; i < quantity; i++) { 
put (genK.next(), genV.next()); 
} 
未 
// A key Generator and a single value: 
public MapData(Generator<k> genK, V value, int quantity){ 
for(int i = @; i < quantity: i++) { 
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put (genK.next(), value): 
} 
} 
// An Iterable and a value Generator: 
public MapData(Iterable<K> genK, Generator<V> genV) { 
for(K key : genK) { 
put(key, genV.next()); 
} 
} 
// An Iterable and a single value: 
public MapData(Iterable<k> genK, V value) { 
for(K key : genK) { 
put(key, value); 
} 
} 
// Generic convenience methods: 
public static <K,V> MapData<K,V> 
map(Generator<Pair<K,V>> gen, int quantity) { 
return new MapData<K,V>(gen, quantity); 
} 
public static <K,V> MapData<K,V> 
map(Generator<K> gen, Generator<V> genV, int quantity) { 
return new MapData<K,V>(genK, genV, quantity); 
} 
public static <K,V> MapDatack,V> 
map(Generator<K> genK, V value, int quantity) { 
return new MapData<k,V>(genK, value, quantity); 
} 
public static <K,V> HapData<K,V> 
map(Iterable<k> genK, Generator<V> genV) { 
return new MapDatacK,V>(genk, genV) ; 
} 
public static <K,V> MapData<K,V> 
map(Iterable<k>, genk, V value) { 
return new MapData<K,V>(genK, value); 
} 
} M~ 


这 给 了 你 一 个 机 会 ， 去 选择 使 用 单一 的 Generator<Pair<K,V>>、 两 个 分 离 的 Generator、 一 个 
Generator 和 一 个 常量 值 、 一 个 Iterable (包括 任何 Collection) 和 一 个 Generator， 还 是 一 个 
Iterable 和 一 个 单一 值 。 泛 型 便利 方法 可 以 减少 在 创建 MapData 类 时 所 必需 的 类 型 检查 数量 。 

下 面 是 一 个 使 用 MapData 的 示例 。LettersGenerator 通 过 产生 一 个 Iterator 还 实现 了 Iterable， 
通过 这 种 方式 ， 它 可 以 被 用 来 测试 MapData.map0 方 法 ， 而 这 些 方法 都 需要 用 到 Tterable: 


//: containers/MapDataTest. java 
import java.util.*; 

import net.mindview.util.*; 

import static net.mindview.util.Print. 


class Letters implements Generator<Pair<Integer,String>>, 
Iterable<Integer> { 
private int size 
private int number = 
private char letter = ‘A 
public Pair<Integer,String> next() { 

return new Pair<Integer, String>( 
number++, "" + letter++); 





$ 
public Iterator<Integer> iterator() { 
return new Iterator<Integer>() { 
public Integer next() { return number++; } 
public boolean hasNext() { return number < size; } 
public void remove() { 
throw new UnsupportedOperationException(); 
} 








797, 
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hi 
} 
} 


public class MapDataTest { 
public static void main(String[] args) { 

// Pair Generator: 

print(MapData.map(new Letters(), 11)); 

// Two separate generators: 

print (MapData.map(new CountingGenerator .Character(), 
new RandonGenerator.String(3). 8)); 

11 A key Generator and a single value: 

print(MapData.map(new Count ingGenerator .Character(). 
"Value", 6)); 

// An Iterable and a value Generator: 

print(MapData.map(new Letters(), 
new RandomGenerator.String(3))): 

// An Iterable and a single value: 

print (MapData.map(new Letters(), "Pop")): 


} 
} /* Output: 
{1=A, 2=B, 3=C, 4=D, 5=E, 6=F, 7 
{a=YNz, b=brn, c=yGc, d=FOW, e=ZnT, 






9=I, 16=J, 115k} 








{aValue, b=Value, c=Value, d: alue, f=Value} 
{1=mJM, 2=RoE, 3=suE, 4=cU0, S=neO, 6=EdL, 7=smw, 8=HLG) 
{1=Pop, 2=Pop, 3=Pop, 4=Pop. 5=Pop, 6=Pop, 7=Pop, 8=Pop} 
Wh 

这 个 示例 也 使 用 了 第 16 章 中 的 生成 器 。 


可 以 使 用 工具 来 创建 任何 用 于 Map 或 Collection 的 生成 数据 集 ， 然 后 通过 构造 器 或 
Map.putAll0 和 Collection.addAll0 方 法 来 初始 化 Map 和 Collection。 

17.2.3 使 用 Abstract 类 ‘ 

对 于 产生 用 于 容器 的 测试 数据 问题 ， 另 一 种 解决 方式 是 创建 定制 的 Colleetion 和 Map 实 现 。 
每 个 java.util 容 器 都 有 其 自己 的 Abstract 类 ， 它 们 提供 了 该 容器 的 部 分 实现 ， 因 此 你 必须 做 的 只 
是 去 实现 那些 产生 想 要 的 容器 所 必需 的 方法 。 如 果 所 产生 的 容器 是 只 读 的 ， 就 像 它 通常 用 的 测 
试 数据 那样 ， 那 么 你 需要 提供 的 方法 数量 将 减少 到 最 少 。 

尽管 在 本 例 中 不 是 特别 需要 ， 但 是 下 面 的 解决 方案 还 是 提供 了 一 个 机 会 来 演示 另 一 种 设计 
模式 享 元 。 你 可 以 在 普通 的 解决 方案 需要 过 多 的 对 象 ， 或 者 产生 普通 对 象 太 占用 空间 时 使 用 
享 元 。 享 元 模式 使 得 对 象 的 一 部 分 可 以 被 具体 化 ， 因 此 ， 与 对 象 中 的 所 有 事物 都 包含 在 对 象 内 
部 不 同 ， 我 们 可 以 在 更 加 高 效 的 外 部 表 中 查找 对 象 的 一 部 分 或 整体 (或 者 通过 某 些 其 他 节省 空 
间 的 计算 来 产生 对 象 的 一 部 分 或 整体 )。 

这 个 示例 的 关键 之 处 在 于 演示 通过 继承 java.util.Abstract 来 创建 定制 的 Map 和 Collection 到 
底 有 多 简单 。 为 了 创建 只 读 的 Map， 可 以 继承 AbstractMap 并 实现 entrySet0。 为 了 创建 只 读 的 
Set， 可 以 继承 AbstractSet 并 实现 iterator0 和 size0。 

本 例 中 使 用 的 数据 集 是 由 世界 上 的 国家 以 及 它们 的 首都 构成 的 Map。 。capitals0 方 法 产生 国 
家 与 首都 的 Map，name0 方 法 产生 国名 的 List。 在 两 种 情况 中 ， 你 都 可 以 通过 提供 表 所 需 尺 寸 的 
int 参 数 来 获取 部 分 列表 : 

//: net/mindview/util/Countries. java 

// “Flyweight” Maps and Lists of sample data. 

package net.mindview. util: 


import java.util.*; 
import static net.mindview.util.Print.*; 


O 这 个 数据 是 从 Internet 上 找到 的 ， 读 者 们 已 经 提交 了 各 种 各 样 的 校正 。 





容器 深入 研究 465 





public class Countries { 
public static final String[][] DATA = { 

1/ Africa 
{"ALGERTA","Algiers")}, {"ANGOLA" ,"Luanda"}, 
{"BENIN" ,"Porto-Novo"}, {"BOTSWANA" ,"Gaberone"}, 
{"BURKINA FASO", "Ouagadougou" }, 
{"BURUNDI" , "Bujumbura" }, 
{"CAMEROON" ,"Yaounde"}, {"CAPE VERDE" ,"Praia"}, 
{"CENTRAL AFRICAN REPUBLIC”, "Bangui"}, 
{"CHAD" ,"N'djamena"}, {"COMOROS", "Horoni"}, 
{"CONGO" ,"Brazzaville"}, {"DJIBOUTI","Dijibouti"}, 
{"EGYPT","Cairo"}, {"EQUATORIAL GUINEA", "Malabo" )}, 
{"ERITREA" ,"Asmara"}, {"ETHIOPIA", "Addis Ababa"), 
{"GABON","Libreville"}, {"THE GAMBIA", "Banjul"}, 
{"GHANA" ,"Accra"}, {"GUINEA","Conakry"}, 
{"BISSAU","Bissau"}, 
{"COTE D'IVOIR (IVORY COAST)" ,"Yamoussoukro"}, 
{"KENYA" ,"Nairobi"}, {"LESOTHO","Maseru"}, 
{"LIBERIA","Monrovia"}, {"LIBYA","Tripoli"}, 
{"MADAGASCAR","Antananarivo"}, {"MALAWI","Lilongwe"), 
{"MALI","Bamako"}, {"MAURITANIA”, "Nouakchott", 
{"MAURITIUS","Port Louis"), {"MOROCCO", “Rabat"}, 
{"MOZAMBIQUE" , "Maputo"}, { "NAMIBIA", "Windhoek"}, 
{"NIGER", "Niamey"}, {"NIGERIA","Abuja"}, 
{"RWANDA" , "Kigali 
{"SAO TOME E PRINCIPE" ,”Sao Tome") 
{"SENEGAL","Dakar"}, {"SEYCHELLES", "Victoria"}, 
{"SIERRA LEONE”, "Freetown"}, {"SOMALIA","Mogadishu"}, 
{"SOUTH AFRICA", "Pretoria/Cape Town"}, 
{"SUDAN", "Khar toum"}, 
{"SWAZILAND","Mbabane"}, {"TANZANIA", "Dodoma"}, 
{"TOGO","Lome"}, {"TUNISIA","Tunis"), 
{"UGANDA" , "Kampala"), 
{"DEMOCRATIC REPUBLIC OF THE CONGO (ZAIRE) ", 
"Kinshasa"), 
{"ZAMBIA","Lusaka"), ("ZIMBABWE*,"Harare"}, 
// Asia 
{"AFGHANISTAN" ,"Kabul"}, { "BAHRAIN", "Manama"), 
{"BANGLADESH","Dhaka"}, {"BHUTAN","Thimphu"}, 
("BRUNEI" "Bandar Seri Begawan"), 
{"CAMBODIA", "Phnom Penh"}, 
{"CHINA", "Bet jing"), {("CYPRUS","Nicosia"), 
{"INDIA", "New Delhi"), {"INDONESIA", "Jakarta", 
{"IRAN","Tehran"}, {"IRAQ","Baghdad"}, 
{"ISRAEL","Jerusalem"), {"JAPAN", "Tokyo"}, 
{"JORDAN","Amman"), {"KUWAIT", "Kuwait City"), 
{"LAOS", "Vientiane"}, {"LEBANON" , "Beirut"} 
{"MALAYSIA", "Kuala Lumpur"}, {"THE MALDIVES: 
("MONGOLIA "Ulan Bator"), 
("MYANMAR (BURMA) ", "Rangoon" }, 
{"NEPAL" , "Katmandu" } , 
{DEMOCRATIC PEOPLE'S REPUBLIC OF KOREA". "P'yongyang"}. 
{"OMAN","Muscat"}, {"PAKISTAN" ,"Islamabad"}, 
{"PHILIPPINES", "Hanila"}, {"QATAR™,"Doha"}, 
{"SAUDI ARABIA", "Riyadh"}, {"SINGAPORE”,"Singapore"), 
("REPUBLIC OF KOREA” ,"Seoul"}, {"SRI LANKA", "Colombo"}, 
{ "SYRIA", “Damascus” 
{ "THAILAND", "Bangkok”}, {"TURKEY", "Ankara"}, 
("UNITED ARAB EMIRATES", "Abu Dhabi"), 
{"VIETNAM", “Hanoi"}, {“YEMEN","Sana‘a"}, 
// Australia and Oceania 
{"AUSTRALIA","Canberra"}, {"FIJI",“Suva"}, 
{"KIRIBATI", "Bairiki"}, 
{"MARSHALL ISLANDS", "Dalap-Uliga-Darrit"}, 
{"MICRONESIA","Palikir"}, {"NAURU™,"Yaren"}, 
{"NEW ZEALAND", "Wellington"}, {"PALAU","Koror"}, 
("PAPUA NEW GUINEA", "Port Moresby"), 
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("SOLOMON ISLANDS", "Honaira"}, {"TONGA", "Nuku'alofa"}, 
{"TUVALU", "Fongafale”}, {"VANUATU","< Port-Vila"}. 
{"WESTERN SAMOA","Apia"}, 

// Eastern Europe and former USSR 

{ "ARMENIA", "Yerevan"}, {"AZERBAIJAN" , "Baku"}, 
("BELARUS (BYELORUSSIA)", “Minsk"}, (“BULGARIA", "Sofia"}, 
{"GEORGIA" , "Tbilisi"}, 

{"KAZAKSTAN" ,"Almaty"}, {"KYRGYZSTAN" ,"Alma-Atā"} , 
{"MOLDOVA", "Chisinau"}, {"RUSSIA","Moscow"}, 

{( "TAJIKISTAN", "Dushanbe"), { "TURKMENISTAN" , "Ashkabad"), 
{"UKRAINE", "Kyiv"}, ("UZBEKISTAN","Tashkent"}, 

// Europe 

{"ALBANIA", "Tirana"}, {"ANDORRA", "Andorra la Vella"}, 
{"AUSTRIA", "Vienna"}, {"BELGIUM","Brussels"}, 
{"BOSNIA","-"}, {"HERZEGOVINA" , "Sarajevo"}, 

{( "CROATIA" ,"Zagreb"}, ("CZECH REPUBLIC", "Prague"), 
{"DENHARK" ,"Copenhagen"}, {"ESTONIA","Tallinn"}, 
{"FINLAND","Helsinki"}, {"FRANCE", "Paris"}, 

{ "GERMANY", "Berlin"}, {"GREECE”,"Athens"}, 

{"HUNGARY™, "Budapest"}, {" ICELAND", "Reykjavik"}, 
{"IRELAND","Dublin"}, {"ITALY","Rome"}, 
{"LATVIA","Riga"), {"LIECHTENSTEIN" , "Vaduz"), 
{"LITHUANIA® ,"Vilnius*}, {"LUXEMBOURG", "Luxembourg"), 
{"MACEDONTA" ,"Skopje"}, ("MALTA","Valletta"}, 
{"MONACO","Monaco"}, {"MONTENEGRO", "Podgorica"}, 

("THE NETHERLANDS”, "Amsterdam"}, {"NORWAY","Oslo"), 
{"POLAND","Warsaw"}, {"PORTUGAL”,"Lisbon"}, 
{"ROMANIA”,"Bucharest"}, {"SAN MARINO", "San Marino"), 

{ "SERBIA", "Belgrade"}, {"SLOVAKIA", "Bratislava"}, 

{ "SLOVENIA", "Ljuijana"}, {"SPAIN","Madrid"), 
{"SWEDEN","Stockholm"}, {"SWITZERLAND”, "Berne"}, 
{"UNITED KINGDOM" ,"London"}, {"VATICAN CITY” 
// North and Central America 

("ANTIGUA AND BARBUDA”,"Saint John's"), 

{ "BAHAMAS", *Nassau"}, 

{ "BARBADOS", "Bridgetown"), {"BELIZE","Belmopan"}, 
{"CANADA" , "Ottawa"}, {"COSTA RICA", "San Jose}, 
{"CUBA","Havana"}, {"DOHINICA","Roseau"}, 

{ "DOMINICAN REPUBLIC", "Santo Domingo"), 

{"EL SALVADOR", "San Salvador"), 

{"GRENADA" "Saint George's"), 

{ "GUATEMALA", "Guatemala City"), 

{"HAITI", "Port-au-Prince"}, 
{"HONDURAS" , "Tegucigalpa"}, {"JAMAICA”,"Kingston"), 
{"MEXICO", "Mexico City"}, {"NICARAGUA™, "Managua"}, 
{"PANAMA", "Panama City"}, {*ST. KITTS A 
{"NEVIS", "Basseterre”"}, {"ST. LUCIA", "Castries"), 
{"ST. VINCENT AND THE GRENADINES", "Kingstown"} 
("UNITED STATES OF AMERICA", "Washington, D.C.*}, 

// South America 

("ARGENTINA", “Buenos Aires"}, 

{"BOLIVIA", "Sucre (legal)/La Paz(administrative)"), 
{"BRAZIL","Brasilia"}, ("CHILE","Santiago"}, 
("COLOMBIA "Bogota"}, {"ECUADOR", "Quito"}, 
{"GUYANA", “Georgetown"}, {"PARAGUAY™, "Asuncion"}, 
{"PERU","Lima"}, {"SURINAME" ,"Paramaribo")}, 
{TRINIDAD AND TOBAGO", "Port of Spain"), 

{ "URUGUAY", "Montevideo"}, {"VENEZUELA™, "Caracas”}, 
































}; 
// Use AbstractMap by implementing entrySet() 
private static class FlyweightMap 
extends AbstractMap<String, String> { 
private static class Entry 
implements Map.Entry<String, String> { 
int index; 
Entry(int index) { this.index = index: } 
public boolean equals (Object o) { 





容器 深入 研究 


467 





return DATA index] [9] .equals(o); 
} 
public String getkey() { return DATA[index] [8]: } 
public String getValue() { return DATALindex] [1]; } 
public String setValue(String value) { 

throw new Unsuppor tedperationException(); 
} 


public int hashCode() { 
return DATA[index] [0] .hashCode() ; 
} 


} 
// Use AbstractSet by implementing size() & iterator() 
static class EntrySet 
extends AbstractSet<Map.Entry<String,String>> { 
private int size; 
EntrySet(int size) { 
if(size < 0) 
this.size = 9; 
// Can't be any bigger than the array: 
else if(size > DATA. length) 
this.size = DATA. length; 
else 
this.size = size; 
} 
public int size() { return size; } 
private class Iter 
implements Iterator<Map.Entry<String,String>> { 
7/ Only one Entry object per Iterator: 
private Entry entry = new Entry(-1); 
public boolean hasNext() { 
return entry.index < size - 
} 
public Map.Entry<String, String> next() { 
entry. index++; 
return entry; 
} 
public void remove() { 
throw new UnsupportedOperationException(); 
} 
} 
public 
Iterator<Map.Entry<String,String>> iterator() { 
return new Iter(); 
》 
} 
private static Set<Map.Entry<String,String>> entries = 
new EntrySet (DATA. Length) ; 
public Set<Map.Entry<String,String>> entrySet() { 
return entries; 
) } 
// Create a partial map of ‘size’ countries: 
static Map<String,String> select(final int size) { 
return new FlyweightMap() { 
public Set<Map.Entry<String,String>> entrySet() { 
return new EntrySet(size); 
» 





和 
} 
static Map<String, String> map = new FlyweightMap(); 
public static Map<String,String> capitals() { 
return map; // The entire map 
$ 
public static Map<String,String> capitals(int size) { 
return select(size); // A partial map 
ti 


static List<String> names = 
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new ArrayList<String>(map.keySet()): 
// ALL the names: 
public static List<String> names() { return names; } 
1/ A partial list: 
public static List<String> names(int size) { 

return new ArrayList<String> (select (size) :keySet()); 


public static void main(String{] args) { 

print(capitals(1@)); 

print(names(10)); 

print (new ‘HashMap<String, String>(capitals(3))): 

print(new LinkedHashMap<Str ing, String> (capitals(3))); 

print(new TreeMap<String, String> (capitals (3))); 

print(new Hashtable<String, String> (capitals(3))); 

print(new HashSet<String>(names(6))); 

print (new LinkedHashSet<String> (names (6))); 

print(new TreeSet<String>(names(6))); 

print(new ArrayList<String>(names(6))); 

print(new LinkedList<String>(names(6))); 

print (capitals().get ("BRAZIL")); 

} 

} /* Output: 
{ALGERIA=Algiers, ANGOLA=Luanda, BENIN=Porto-Novo, 
BOTSWANA=Gaberone, BULGARIA=Sofia, BURKINA 
FASO=Quagadougou, BURUNDI=Bujumbura, CAMEROON=Yaounde, CAPE 
VERDE=Praia. CENTRAL AFRICAN REPUBLIC=Bangui} 
(ALGERIA, ANGOLA, BENIN, BOTSWANA, BULGARIA, BURKINA FASO, 
BURUNDI, CAMEROON, CAPE VERDE, CENTRAL AFRICAN REPUBLIC] 
{BENIN=Porto-Novo, ANGOLA=Luanda, ALGERIA=Algiers) 
{ALGERIA=Algiers, ANGOLA=Luanda, BENIN=Porto-Novo} 
{ALGERIA=Algiers, ANGOLA=Luanda, BENIN=Porto-Novo} 
{ALGERIA=Algiers, ANGOLA=Luanda, BENIN=Porto-Novo} 
[BULGARIA, BURKINA FASO, BOTSWANA, BENIN, ANGOLA, ALGERIA] 
(ALGERIA, ANGOLA, BENIN, BOTSWANA, BULGARIA, BURKINA FASO] 
TALGERIA, ANGOLA, BENIN, BOTSWANA, BULGARIA, BURKINA FASO] 
TALGERIA, ANGOLA, BENIN. BOTSWANA, BULGARIA, BURKINA FASO] 
TALGERIA, ANGOLA, BENIN, BOTSWANA, BULGARIA, BURKINA FASO] 
Brasilia 
“i~ 


二 维 数组 String DATA 是 public 的 ， 因 此 它 可 以 在 其 他 地 方 使 用 。FlyweightMap 必 须 实 
现 entrySet() 方 法 ， 它 需要 定制 的 Set 实 现 和 定制 的 Map.Entry 类 。 这 里 正 是 享 元 部 分 : 每 个 
Map.Entry 对 象 都 只 存储 了 它 的 索引 ， 而 不 是 实际 的 键 和 值 。 当 你 调用 getKeyO 和 getValue0) 
时 ， 它 们 会 使 用 该 索引 来 返回 恰当 的 DATA 元 素 。EntrySet 可 以 确保 它 的 size 不 会 大 于 
DATA, 

你 可 以 在 EntrySet.Iterator 中 看 到 享 元 其 他 部 分 的 实现 。 与 为 DATA 中 的 每 个 数据 对 都 创建 
Map.Entry 对 象 不 同 ， 每 个 选 代 器 只 有 一 个 Map.Entry。Entry 对 象 被 用 作 数 据 的 视窗 ， 它 只 包 
含 在 静态 字符 串 数组 中 的 索引 。 你 每 次 调用 迭代 器 的 next0 方 法 时 ，Entry 中 的 index 都 会 递增 ， 
使 其 指向 下 一 个 元 素 对 ， 然 后 从 next0 返 回 该 terator 所 持 有 的 单一 的 Entry 对 象 。 

select() 方 法 将 产生 一 个 包含 指定 尺寸 的 EntrySet 的 FlyweightMap， 它 会 被 用 于 重 载 过 的 
capitals0 和 names0 方 法 ， 正 如 在 main0 中 所 演示 的 那样 。 

对 于 某 些 测试 ，Countries 的 尺寸 受 限 会 成 为 问题 。 我 们 可 以 采用 与 产生 定制 容器 相同 的 方 
式 来 解决 ， 其 中 定制 容器 是 经 过 初始 化 的 ， 并 且 具 有 任意 尺寸 的 数据 集 。 下 面 的 类 是 一 个 List， 
它 可 以 具有 任意 尺寸 ， 并 且 用 Integer 数 据 (有 效 地 ) 进行 了 预 初始 化 : 、 


日 javautil 中 的 Map 使 用 了 用 于 Map 的 getKey0 和 getValue0 来 执行 成 批复 制 ， 因 此 这 样 是 可 以 工作 的 。 如 果 定 制 
的 Map 直 接 复制 完整 的 Map.Entry， 那 么 这 种 方法 就 会 有 问题 。 
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//: net/mindview/util/CountingIntegerList.java 
7/ List of any length, containing sample data. 
package net.mindview.util; 

import java.util.*; 


public class CountingIntegerList 
extends AbstractList<Integer> { 
private int size; 
public CountingIntegerList(int size) { 
this.size = size < @ ? @ : size: 
} 
public Integer get(int index) { 
return Integer .value0f (index); 
} 
public int size() { return size; } 
public static void main(String[] args) { 
System.out.printin(new CountingIntegerList(30)); 
} 
} /* Output: 
(0, 2, 2, 3. 4, 5, 6, 7, 8, 9, 10, 11, 12, 13, 14, 15, 16, 
17, 18, 19, 20, 21, 22, 23, 24, 25, 26, 27, 28, 29] 
Wha 


为 了 从 AbstractList 创 建 只 读 的 List， 你 必须 实现 get0 和 size0。 这 里 再 次 使 用 了 享 元 解决 方 
当 你 寻找 值 时 ，get0 将 产生 它 ， 因 此 这 个 List 实 际 上 并 不 必 组 装 。 
下 面 是 包含 经 过 预 初始 化 ， 并 且 都 是 唯一 的 Integer 和 String 对 的 Map， 它 可 以 具有 任意 尺寸 ; 


/1/: net/mindview/util/CountingMapData. java 

// Unlimited-Length Map containing sample data. 
package net.mindview.util; 

import java.util.*; 





public class Count ingMapData 
extends AbstractMap<Integer String> { 
private int size; 
private static String{] chars = 
"“ABCDEFGHIJKLMNOPQRSTUVWXY Z" 
ssplit(" "); 
public CountingMapData(int size) { 
if (size < 0) this.size = @; 
this.size = size; 
} 
private static class Entry 
implements Map.Entry<Integer,String> { 
int index; 
Entry(int index) { this. index = index; } 
public boolean equals(Object o) { 
return Integer .value0f (index) equals (o) ; 
} 
public Integer getKey() { return index: } 
public String getValue() { 
return 
chars[index % chars.length] + 
Integer. toString(index / chars.length); 


} 
public String setValue(String value) { 
throw new Unsuppor tedOperat ionExcept ion(); 
A 
public int hashCode() { 
return Integer .value0f (index) .hashCode() ; 
》 
} 
public Set<Map.Entry<Integer,String>> entrySet() { 
// LinkedHashSet retains initialization order: 
_Set<Map.Entry<Integer,String>> entries = 
new LinkedHashSet<Map. Entry<Integer, String>>(); 
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for(int i =; i < size; i+*) 
entries.add(new Entry(i)); 
return entries; 


} 
public static void main(String[] args) { 
System.out.printin(new CountingMapData(6@)) ; 


} /* Output: 

{9=A9, 1=B0, 2=CO, 3=D0, 4=E0, S=FO, 6=G9, 7=HO, 
9=J@, 10=K9, 11=L@, 12=M9, 1 08 

17=R8, 18=S0, 19=TO, 20=U0, 
25=20, 26=Al, 27=B1, 28=C1, i 
33=H1, 34=11, 35=J1, 36=K1, 37=L1 
41=P1, 42=Q1, 43=R1, 44=S1, 45=T1, ,47=V1, 
49=X1, 50=Y1, 51=Z1, 52=A2, 53=B2, 54=C2, 55=D2, 
57=F2, 58=G2, 59=H2} 

“Ii~ 


这 里 使 用 的 是 LinkedHashSet， 而 不 是 定制 的 Set 类 ， 因 此 享 元 并 未 完全 实现 。 

练习 1: (1) 创建 一 个 List (用 ArrayList 和 LinkedList 都 尝试 一 下 ) ， 然 后 用 Countries 来 填充 。 
对 该 列表 排序 并 打印 ,然后 将 Collections.shuffle0 方 法 重复 地 应 用 于 该 列表 , 并 且 每 次 都 打印 它 ， 
这 样 你 就 可 以 看 到 shuffle0 方 法 是 如 何 每 次 都 将 列表 随机 打 乱 的 了 。 

练习 2: (2) 生成 一 个 Map 和 Set， 使 其 包含 所 有 以 字母 A 开 头 的 国家 。 

练习 3: (1) 使 用 Countries， 用 同样 的 数据 多 次 填充 Set， 然 后 验证 此 Set 中 没有 重复 的 元 素 。 
使 用 HashSet、LinkedHashSet 和 TreeSet 做 此 测试 。 

练习 4: (2) 创建 一 个 Collection 初 始 化 器 , 它 将 打开 一 个 文件 , 并 用 TextFile 将 其 断 开 为 单词 ， 
然后 将 这 些 单词 作为 所 产生 的 Collection 的 数据 源 使 用 。 请 演示 它 是 可 以 工作 的 。 

练习 5: (3) 修改 CountingMapDatajava， 通 过 添加 像 Countriesjava 中 那样 的 定制 EntrySet 
类 ， 来 完全 实现 享 元 。 
17.3 _ Collection 的 功能 方法 

下 面 的 表格 列 出 了 可 以 通过 Colleetion 执 行 的 所 有 操作 (不 包括 从 Object 继承 而 来 的 方法 ) 。 
因此 ， 它 们 也 是 可 通过 Set 或 List 执 行 的 所 有 操作 (List 还 有 额外 的 功能 ) 。Map 不 是 继承 自 
Collection 的 ， 所 以 会 另行 介绍 。 

boolean add(T) 确保 容器 持 有 具有 泛 型 类 型 T 的 参数 。 如 果 没 有 将 此 参数 添加 进 容 


器 ， 则 返回 false (这 是 “可 选 ”的 方法 ， 稍 后 会 解释 ) 
boolean addAll(Collection<? extends T>) 添加 参数 中 的 所 有 元 素 。 只 要 添加 了 任意 元 素 就 返回 true (可 选 的 ) 








void clear() 移 除 容器 中 的 所 有 元 素 (TAR) 

boolean contains(T) 如 果 容 器 已 经 持 有 具有 泛 型 类 型 T 此 参数 ， 则 返回 true 

Boolean containsAll(Collection<?>) 如 果 容器 持 有 此 参数 中 的 所 有 元 素 ， 则 返回 rue 

boolean isEmpty() 容器 中 没有 元 素 时 返回 true 

Iterator<T> iterator() 返回 一 个 Iterator<T>， 可 以 用 来 遍历 容器 中 的 元 素 

Boolean remove(Object) 如 果 参 数 在 容器 中 ， 则 移 除 此 元 素 的 一 个 实例 。 如 果 做 了 移 除 动作 ， 
则 返回 rue( 可 选 的 ) 

boolean removeAll(Collection<?>) 移 除 参数 中 的 所 有 元 素 。 只 要 有 移 除 动作 发 生 就 返回 true (可 选 的 ) 

Boolean retainAll(Collection<?>) 只 保存 参数 中 的 元 素 (应 用 集合 论 的 “交集 ”概念 ) 。 只 要 
Collection 发 生 了 改变 就 返回 true (可 选 的 ) 

int size) 返回 容器 中 元 素 的 数目 

ObjectD toArray0 返回 一 个 数组 ， 该 数组 包含 容器 中 的 所 有 元 素 

<T> TO toArray(T[] a) 返回 一 个 数组 ， 该 数组 包含 容器 中 的 所 有 元 素 。 返 回 结 果 的 运行 时 


类 型 与 参数 数组 a 的 类 型 相同 ， 而 不 是 单纯 的 Object 
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请 注意 ， 其 中 不 包括 随机 访问 所 选择 元 素 的 get0 方 法 。 因 为 Collection 包 括 Set， 而 Set 是 自 
己 维 护 内 部 顺序 的 〈 这 使 得 随机 访问 变 得 没有 意义 )。 因 此 ， 如 果 想 检查 Collection 中 的 元 素 ， 
那 就 必须 使 用 迭代 器 。 

下 面 的 例子 展示 了 所 有 这 些 方 法 。 虽 然 任 何 实现 了 Collection 的 类 都 可 以 使 用 这 些 方法 ,但 
示例 中 使 用 ArrayList， 以 说 明 各 种 Collection 子 类 的 “最 基本 的 共同 特性 ”: 


//: containers/CollectionMethods.java 

// Things you can do with all Collections. 
import java.util.*; 

import net.mindview.util.*; 

import static net.mindview.util.Print.*: 





public class CollectionMethods { 
public static void main(String[] args) { 
Collection<String> c = new ArrayList<String>(); 
c.addAll (Countries names (6)) ; 
c.add("ten"); 
c.add ("eleven"); 
print(c); 
// Make an array from the List: 
Object[] array = c.toArray(): 
1/ Make a String array from the List: 
String] str = c.toArray(new String[6]); 
// Find max and min elements; this means 
// different things depending on the way 
71 the Comparable interface is implemented: 
print("Collections.max(c) = ”+ Collections.max(c) 
print(*Collections.min(c) = " + Collections.min(c)); 
// Add a Collection to another Collection 
Cotlection<String> c2 = new Arraylist<String>(): 
c2.addAll (Countries names (6)) ; 
¢.addAL1(€2) ; 
print(c); 
c. remove (Countries .DATA{@] [9]); 
print(c); 
¢. remove (Countries. DATA(1) [9] ); 
print(c); 
7/ Remove all components that are 811 
// in the argument collection: 
©. removeAll (c2); 
print(c); 
€.addAll (c2); 
print(c); 
// Is an element in this Collection? 
String val = Countries .DATA(3) [8 
Print("c.contains(* + val +") =" + c.contains(val)); 
// Is a Collection in this Collection? 
print (“c.containsall(c2) = * + c.containsAtl(c2)); 
Collection<String> c3 = 
((List<String>)c).sublist(3, 5); 
// Keep all the elements that are in both 
// c2 and c3 (an intersection of sets): 
€2.retainAll (c3); 
print(c2); 
// Throw away all the elements 
// in c2 that also appear in c3: 
c2. removeAll (c3); 
print("c2.isEmpty() = ”+ c2.1sEmpty()); 
c = new Arraylist<String>(); 
c.addAl1 (Countries names (6)); 
print(c): 
c.clear(); // Remove all elements 
print("after c.clear():" + ¢); 
} 
) /* Output: 
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` [ALGERTA, ANGOLA, BENIN, BOTSWANA, BULGARIA, BURKINA FASO, 
ten, eleven] 
Collections.max(c) = ten 
Collections.min(c) = ALGERIA 
{ALGERIA, ANGOLA, BENIN, BOTSWANA, BULGARIA, BURKINA FASO, 
ten, eleven, ALGERIA, ANGOLA, BENIN, BOTSWANA, BULGARIA, 
BURKINA FASO] 
[ANGOLA, BENIN, BOTSWANA, BULGARIA, BURKINA FASO, ten, 
eleven, ALGERIA, ANGOLA, BENIN, BOTSWANA, BULGARIA, BURKINA 
FASO} 
[BENIN, BOTSWANA, BULGARIA, BURKINA FASO, ten, eleven, 
ALGERIA, ANGOLA, BENIN, BOTSWANA, BULGARIA, BURKINA FASO] 
[ten, eleven] 
[ten, eleven, ALGERIA, ANGOLA, BENIN, BOTSWANA, BULGARIA, 
BURKINA FASO] 
c.contains (BOTSWANA) = true 
c.containsAll (c2) = true 
(ANGOLA, BENIN] 
c2.isémpty() = true 
(ALGERIA, ANGOLA, BENIN, BOTSWANA, BULGARIA, BURKINA FASO] 
after c.clear():[) 
Whew 


创建 ArrayList 来 保存 不 同 的 数据 集 ， 然 后 向 上 转型 为 Collection， 所 以 很 明显 ， 代 码 只 用 到 
了 Collection 接 口 。main0 用 简单 的 练习 展示 了 Collection 中 的 所 有 方法 。 

本 章 后 面 的 小 节 将 介绍 List、Set 和 Map 的 各 种 实现 ， 每 种 情况 都 会 〈 以 星 号 ) 标 出 默认 的 
选择 。 对 遗留 类 Vector、Stack 和 Hashtable 的 描述 放 到 了 本 章 的 末尾 ， 尽 管 你 不 应 该 使 用 这 些 类 ， 
但 是 在 老 的 代码 中 仍 就 会 看 到 它们 。 


17.4 可 选 操作 


执行 各 种 不 同 的 添加 和 移 除 的 方法 在 Collection 接 口中 都 是 可 选 操作 。 这 意味 着 实现 类 并 不 
需要 为 这 些 方法 提供 功能 定义 。 

这 是 一 种 很 不 寻常 的 接口 定义 方式 。 正 如 你 所 看 到 的 那样 ， 接 口 是 面 向 对 象 设 计 中 的 契约 ， 
它 声明 “无 论 你 选择 如 何 实现 该 接口 ， 我 保证 你 可 以 向 该 接口 发 送 这 些 消息 ? 。” 但 是 可 选 操作 
违反 这 个 非常 基本 的 原则 ， 它 声明 调用 某 些 方法 将 不 会 执行 有 意义 的 行为 ， 相 反 ， 它 们 会 抛 出 
异常 。 这 看 起 来 好 像 是 编译 期 类 型 安全 被 抛弃 了 ! 

事情 并 不 那么 精 。 如 果 一 个 操作 是 可 选 的 ， 编 译 器 仍旧 会 严格 要 求 你 只 能 调用 该 接口 中 的 
方法 。 这 与 动态 语言 不 同 ， 动 态 语言 可 以 在 任何 对 象 上 调用 任何 方法 ， 并 且 可 以 在 运行 时 发 现 
某 个 特定 调用 是 否 可 以 工作 。。 另 外 ， 将 Collection 当 作 参 数 接受 的 大 部 分 方法 只 会 从 该 
Collection 中 读 取 ， 而 Collection 的 读 取 方法 都 不 是 可 选 的 。 

为 什么 你 会 将 方法 定义 为 可 选 的 呢 ? 那 是 因为 这 样 做 可 以 防止 在 设计 中 出 现 接口 爆炸 的 情 
况 。 容 器 类 库 中 的 其 他 设计 看 起 来 总 是 为 了 描述 每 个 主题 的 各 种 变 体 ， 而 最 终 患 上 了 令 人 困惑 
的 接口 过 剩 症 。 甚 至 这 么 做 仍 不 能 捕捉 接口 的 各 种 特例 ， 因 为 总 是 有 人 会 发 明 新 的 接口 。“ 未 获 
支持 的 操作 ”这 种 方式 可 以 实现 Java 容 器 类 库 的 一 个 重要 目标 : 容器 应 该 易学 易 用 。 未 获 支持 
的 操作 是 一 种 特例 ， 可 以 延迟 到 需要 时 再 实现 。 但 是 ， 为 了 让 这 种 方式 能 够 工作 ， 

1. UnsupportedOperationException 必 须 是 一 种 罕见 事件 。 即 ， 对 于 大 多 数 类 来 说 ， 所 有 操 


作 都 应 该 可 以 工作 ， 只 有 在 特例 中 才 会 有 未 获 支持 的 操作 。 在 Java 容 器 类 库 中 确实 如 此 ， 因 为 


O 我 在 这 里 使 用 术语 “接口 ”来 描述 正式 的 interface 关 键 字 和 “任何 类 或 子 类 支持 的 方法 ”这 一 更 通用 的 含义 。 
O 尽管 当 我 以 这 种 方式 来 描述 时 ， 听 起 来 会 感觉 很 奇怪 ， 并 且 显 得 有 些 无 用 ， 但 是 正如 你 所 看 到 的 ， 特 别 是 在 
第 14 章 中 ， 这 种 类 型 的 动态 行为 会 显得 非常 强大 。 


I 
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你 在 99% 的 时 间 里 面 使 用 的 容器 类 ， 如 ArrayList、LinkedList、HashSet 和 HashMap， 以 及 其 他 
的 具体 实现 ， 都 支持 所 有 的 操作 。 这 种 设计 留 下 了 一 个 “后 门 "， 如 果 你 想 创建 新 的 Collection ， 
但 是 没有 为 Collection 接 口中 的 所 有 方法 都 提供 有 意义 的 定义 ， 那么 它 仍旧 适合 现 有 的 类 库 。 

2. 如 果 一 个 操作 是 未 获 支持 的 ， 那 么 在 实现 接口 的 时 候 可 能 就 会 导致 UnsupportedOperation- 
Exception 异常 ， 而 不 是 将 产品 程序 交 给 客户 以 后 才 出 现 此 异常 ， 这 种 情况 是 有 道理 的 。 毕 竞 ， 它 
表示 编程 上 有 错误 : 使 用 了 不 正确 的 接口 实现 。 

值得 注意 的 是 ， 未 获 支持 的 操作 只 有 在 运行 时 才能 探测 到 ， 因 此 它们 表示 动态 类 型 检查 。 
如 果 你 以 前 使 用 的 是 像 C++ 这 样 的 静态 类 型 语言 ， 那 么 可 能 会 觉得 Java 也 只 是 另 一 种 静态 类 型 语 
言 ， 但 是 它 还 具有 大 量 的 动态 类 型 机 制 ， 因 此 很 难说 它 到 底 是 哪 一 种 类 型 的 语言 。 一 旦 开始 注 
意 到 这 一 点 了 ， 你 就 会 看 到 Java 中 动态 类 型 检查 的 其 他 例子 。 


17.4.1 未 获 支持 的 操作 

最 常见 的 未 获 支持 的 操作 ， 都 来 源 于 背后 由 固定 尺寸 的 数据 结构 支持 的 容器 。 当 你 用 
Arrays.asList0 将 数组 转换 为 List 时 ， 就 会 得 到 这 样 的 容器 。 你 还 可 以 通过 使 用 Collections 类 中 
“不 可 修改 ”的 方法 ， 选 择 创建 任何 会 抛 出 UnsupportedOperationException 的 容器 (包括 Map)。 
下 面 的 示例 包括 这 两 种 情况 : 


//: containers/Unsupported. java 
// Unsupported operations in Java containers. 
import java.util.*; 


public class Unsupported { 
static void test(String msg, List<String> list) { 
System.out.printin("--- " + msg + “ ---"): 
Cotlection<String> c = list; 
Collection<String> subList = List.subList(1,8); 
// Copy of the sublist: 
Collection<String> c2 = new ArrayList<String>(subList); 
try { c.retaimAll(c2); } catch(Exception e) { 
System.out.printin("retainAll(): ”+ e); 


try { c.removeAll(c2); } catch(Exception e) { 
‘System.out.printin("removeAll(): ”+ e); 


try { c.clear(); } catch(Exception e) { 
System. out.printin("clear(): ”+ e); 


} 
try { c.add("X"); } catch(Exception e) { 
System.out.printin("add(): * + e); 


} 

try { c.addAll(c2); } catch(Exception e) { 
System.out.printin("addAll(): ”+ e); 

} 

try { €.remove("C"); } catch(Exception e) { 
System.out.printin("remove(): ”+ e); 


} 
// The List.set() method modifies the value but 
// doesn't change the size of the data structure: 
try { 
List.set(®, "X"); 
} catch(Exception e) { 
System.out.printin("List.set(): " + e); 
) 
} 
public static void main(String{] args) { 
List<String> list = 
Arrays.asList("ABCDEFGHI JK L".split(" ")); 
test("Modifiable Copy", new ArrayList<String> (list); 
test("Arrays.asList()", list); 
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test("unmodifiableList()", 
Collections.unmodifiableList( 
new ArrayList<String>(list))); 
} 
} /* Output: 
-- Modifiable Copy --- 
--- Arrays.asList() --- 
retainAll(): java. lang.UnsupportedOperationException 
removeAll(): java.1ang.UnsupportedOperat ionException 
clear(): java.lang.Unsuppor tedOperationExcept ion 
add(): java. lang.UnsupportedOperat ionExcept ion 
addAll(): java. lang.Unsuppor tedOperationExcept ion 
remove(): java.lang.UnsupportedOperat ionException 
-- unmodifiableList() --- 
retainAll(): java. lang.UnsupportedOperationException 
removeAll(): java.lang.UnsupportedOperationException 
clear(): java. lang.UnsupportedOperat ionException 
add(): java. lang .Unsuppor tedOperat ionExcept ion 
addA11(): java. lang.UnsupportedOperation€xcept ion 
remove(): java. lang.UnsupportedOperat ionException 
List.set(): java. lang.Unsuppor tedOperat ionExcept ion 
Wim 


因为 Arrays.asList0 会 生成 一 个 List， 它 基于 一 个 固定 大 小 的 数组 ， 仅 支持 那些 不 会 改变 数组 
大 小 的 操作 ， 对 它 而 言 是 有 道理 的 。 任 何 会 引起 对 底层 数据 结构 的 尺寸 进行 修改 的 方法 都 会 产生 
一 个 UnsupportedOperationException 异 常 ， 以 表示 对 未 获 支持 操作 的 调用 (一 个 编程 错误 )。 

注意 ， 应 该 把 Arrays.asList() 的 结果 作为 构造 器 的 参数 传递 给 任何 Collection (或 者 使 用 
addAll0 方 法 ， 或 Collections.addAll0 静 态 方法 )， 这 样 可 以 生成 允许 使 用 所 有 的 方法 的 普通 容 
器 一 这 在 main0) 中 的 第 一 个 对 test0 的 调用 中 得 到 了 展示 ， 这 样 的 调用 会 产生 新 的 尺寸 可 调 的 
底层 数据 结构 。Collections 类 中 的 “不 可 修改 ”的 方法 将 容器 包装 到 了 一 个 代理 中 ， 只 要 你 执 
行 任何 试图 修改 容器 的 操作 ， 这 个 代理 都 会 产生 UnsupportedOperationException 异 常 。 使 用 这 
些 方法 的 目标 就 是 产生 “常量 ”容器 对 象 。“ 不 可 修改 ”的 Collections 方 法 的 完整 列表 将 在 稍 后 
介绍 。 

test0 中 的 最 后 一 个 try 语 句 块 将 检查 作为 List 的 一 部 分 的 set() 方 法 。 这 很 有 趣 ， 因 为 你 可 以 
看 到 “未 获 支持 的 操作 ”这 一 技术 的 粒度 来 的 是 多 么 方便 一 -所 产生 的 “接口 ”可 以 在 
Arrays.asList0) 返 回 的 对 象 和 Collections.unmodifiableList0 返 回 的 对 象 之 间 ， 在 一 个 方法 的 粒 
度 上 产生 变化 。Arrays.asList0 返 回 固定 尺寸 的 List， 而 Collections.unmodifiableList0 产 生 不 可 
修改 的 列表 。 正 如 从 输出 中 所 看 到 的 ， 修 改 Arrays.asList0 返 回 的 List 中 的 元 素 是 可 以 的 ， 因 为 
这 没有 违反 该 List“ 尺 寸 固定 ”这 一 特性 。 但 是 很 明显 ，unmodifiableList0 的 结果 在 任何 情况 
下 都 应 该 不 是 可 修改 的 。 如 果 使 用 的 是 接口 ， 那 么 还 需要 两 个 附加 的 接口 ， 一 个 具有 可 以 工作 
的 set0 方 法 ， 另 外 一 个 没有 ， 因 为 附加 的 接口 对 于 Collection 的 各 种 不 可 修改 的 子 类 型 来 说 是 必 
需 的 。 

对 于 将 容器 作为 参数 接受 的 方法 ， 其 文档 应 该 指定 哪些 可 选 方法 必须 实现 。 

练习 6: (2) 注意 ，List 具 有 附加 的 “可 选 ”操作 ， 它 们 不 包含 在 Colleetion 中 。 编 写 一 个 
Unsupportedjava 版 本 ， 测 试 这 些 附加 的 可 选 操作 。 


17.5 List 的 功能 方法 


正如 你 所 看 到 的 ， 基 本 的 List 很 容易 使 用 : 大 多 数 时 候 只 是 调用 add0 添 加 对 象 ， 使 用 get0 
一 次 取出 一 个 元 素 ， 以 及 调用 iterator0 获 取 用 于 该 序列 的 Iterator。 
下 面 例 子 中 的 每 个 方法 都 涵盖 了 一 组 不 同 的 动作 ，basicTest0 中 包含 每 个 List 都 可 以 执行 的 
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操作 ，iterMotion0 使 用 Iterator 人 遍历 元 素 ， 对 应 的 iterManipulation0 使 用 Iterator 修 改元 素 ， 
testVisual0 用 以 查看 List 的 操作 效果 ， 还 有 一 些 LinkedList 专 用 的 操作 。 


//: containers/Lists.java 
// Things you can do with Lists. 

import java.util.*; 

import net.mindview.util.*; 

import static net.mindview.util.Print.*; 


public class Lists { ; 
private static boolean b; 
private static String 
private static int 1; 
private static Iterator<String> it; 
private static ListIterator<String> lit; 
public static void basicTest(List<String> a) { 
a.add(1, "x"); // Add at location 1 
a.add("x"): // Add at end 
// Add a collection: 
a.addA1l (Countries. names(25)); 
11 Add a collection starting at location 3: 
a.addA11(3, Countries.names(25)); 
b = a.contains("1"); // Is it in there? 
// Is the entire collection in there? 
b = a.containsAll(Countries.names(25)); 
// Lists allow random access, which is cheap 
// for ArrayList, expensive for LinkedList: 
s = a.get(1); // Get (typed) object at location 1 
1 = a.indexOf("1"); // Tell index of object 
b = a.isEmpty(); // Any elements inside? 
it = a.iterator(); // Ordinary Iterator 
lit = a.listIterator(); // ListIterator 
Ut = a.listIterator(3); // Start at loc 3 
1 = a.lastIndexOf("1"); // Last match 
a.remove(1); // Remove location 1 
a.remove("3"); // Remove this object 
a.set(1, "y"); // Set location 1 to "y" 
// Keep everything that's in the argument 
17 (the intersection of the two sets): 
a.retainAll (Countries.names(25)); 
// Remove everything that's in the argument: 
a. removeAll (Countries. names (25)) ; 
i = a.size(); // How big is it? 
a.clear(); // Remove all elements 
} 
public static void iterMotion(List<String> a) { 
ListIterator<String> it = a.listIterator(); 
= it-hasNext() ; 
= it.hasPrevious(); 
= itenext(); 
it.nextIndex(); 
it.previous(); 
it.previousindex() ; 











wangogo 


} 

Public static void iterManipulation(List<String> a) { 
ListIterator<String> it = a.listIterator(); 
it.add("47"); 

// Must move to an element after add(): 

it.nextQ); 

// Remove the element after the newly produced one: 
it. remove(); 

// Must move to an element after remove(): 
it.next(); 

// Change the element after the deleted one: 
it.set("47"); 
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public static void testVisual(List<String> a) { 
print(a): 
List<String> b = Countries.names (25); 
print("b = ”+ b); 
aaddAtL(b 
aaddALL(b) ; 
print(a); 
// Insert, remove, and replace elements 
// using a ListIterator: 
ListIterator<String> x = a.listIterator(a.size()/2); 
x.add("one"); A 
print(a); 
print(x.next()); 
x. remove(); 
print(x.next()): 
x. set ("47"); 
print(a); 
// Traverse the list backwards: 
x = a. listIterator(a.size()); 
while(x.hasPrevious()) 

printnb(x.previous() + " "); 

print(); 
print("testVisual finished"); 











NA 
// There are some things that only LinkedLists can do: 


public static void testLinkedList() { 


LinkedList<String> 11 = new LinkedList<String>(); 
UL. addAll (Countries names (25) ) ; 
print(tl); 


// Treat it like a stack, pushing: 
ll. addFirst("one" 
LlL.addFirst ("two") ; 
print(ti); 
// Like "peeking" at the top of a stack: 
print (1l.getFirst()): 
// Like popping a staci 
print(1l.removeFirst() 
print(11.removeFirst()); 
// Treat it like a queue, pulling elements 
// off the tail end: 
print(1l.removelast()); 
print(1t): 
} 
Public static void main(String[] args) { 

// Make and fill a new list each time: 
basicTest( 

new LinkedList<String>(Countries.names(25))); 
basicTest( 

new ArrayList<String>(Countries.names(25))); 
iterMotion( 

new LinkedList<String>(Countries.names(25))); 
iterMotion( 

new ArrayList<String>(Countries.names(25))); 
itertanipulation( 

new LinkedList<String> (Countries .names(25))); 
iterManipulation( 

new ArrayList<String> (Countries .names(25))) ; 
testVisual ( 

new LinkedList<String>(Countries.names(25))); 
testLinkedlist(); 








} 
} /* (Execute to see output) *///:~ 


在 basicTest0 和 iterMotion0 方 法 中 ， 调 用 只 是 为 了 演示 正确 的 语法 ， 虽 然 取得 了 返回 值 ， 
却 没有 使 用 。 某 些 情 况 则 根本 没有 捕获 返回 值 。 使 用 这 些 方法 前 ， 应 该 查询 JDK 帮 助 文档 ， 以 
充分 了 解 各 种 方法 的 用 途 。 
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练习 7: (4) 分 别 创建 一 个 ArrayList 和 LinkedList， 用 Countries.names0 生 成 器 来 填充 每 个 
容器 。 用 普通 的 Iterator 打 印 每 个 列表 ， 然 后 用 ListIterator 按 隔 一 个 位 置 插入 一 个 对 象 的 方式 把 
一 个 表 插入 到 另 一 个 表 中 。 现 在 ， 从 第 1 个 表 的 末尾 开始 ， 向 前 移动 执行 插入 操作 。 

练习 8: (7) 创建 一 个 泛 型 的 单 向 链表 类 SList， 为 了 简单 起 见 ， 不 要 让 它 去 实现 List 接 口 。 
列表 中 的 每 个 Link 对 象 都 应 该 包含 一 个 对 列表 中 下 一 个 元 素 而 不 是 前 一 个 元 素 的 引用 (与 这 个 
类 相 比 ，LinkedList 是 双向 链表 ， 它 包含 两 个 方向 的 链接 ) 。 创 建 你 自己 的 SListIterator， 同 样 
为 了 简单 起 见 ， 不 要 实现 List[terator。SList 中 除了 toString0 之 外 唯一 的 方法 应 该 是 iterator0 ， 
它 将 产生 一 个 SListIterator。 在 SList 中 插入 和 移 除 元 素 的 唯一 方式 就 是 通过 SListIterator。 编 写 
代码 来 演示 SList。 


17.6 Set 和 存储 顺序 


在 第 11 章 中 的 Set 示 例 对 可 以 用 基本 的 Set 执 行 的 操作 ， 提 供 了 很 好 的 介绍 。 但 是 那些 示例 很 
方便 地 使 用 了 诸如 Integer 和 String 这 样 的 Java 预 定义 的 类 型 ， 这 些 类 型 被 设计 为 可 以 在 容器 内 部 
使 用 。 当 你 创建 自己 的 类 型 时 ， 要 意识 到 Set 需 要 一 种 方式 来 维护 存储 期 序 ， 而 存储 顺序 如 何 维 
护 ， 则 是 在 Set 的 不 同 实现 之 间 会 有 所 变化 。 ， 不 同 的 Set 实 现 不 仅 具有 不 同 的 行为 ， 而 且 它 
们 对 于 可 以 在 特定 的 Set 中 放置 的 元 素 的 类 型 也 有 不 同 的 要 求 ， 








Set (interface) 存 入 Set 的 每 个 元 素 都 必须 是 唯一 的 ， 因 为 Set 不 保存 重复 元 素 。 加 入 Set 的 元 素 必须 定义 
equals0 方 法 以 确保 对 象 的 唯一 性 。Set 与 Collection 有 完全 一 样 的 接口 。Set 接 口 不 保证 维护 元 
素 的 次 序 

HashSet* 为 快速 查找 而 设计 的 Set。 存 人 HashSet 的 元 素 必 须 定义 hashCode0 

‘TreeSet 保持 次 序 的 Set， 底 层 为 树 结构 。 使 用 它 可 以 从 Set 中 提取 有 序 的 序列 。 元 素 必 须 实现 
Comparable 接 品 

LinkedHashSet 具有 HashSet 的 查询 速度 ， 且 内 部 使 用 链表 维护 元 素 的 顺序 (插入 的 次 序 )。 于 是 在 使 用 达 


代 器 遍历 Set 时 ， 结 果 会 按 元 素 插入 的 次 序 显示 。 元 素 也 必须 定义 hashCode() 方 法 


在 HashSet 上 打 星 号 表示 ， 如 果 没 有 其 他 的 限制 ， 这 就 应 该 是 你 默认 的 选择 ， 因 为 它 对 速度 
进行 了 优化 。 

定义 hashCode0 的 机 制 将 在 本 章 稍 后 进行 介绍 。 你 必须 为 散 列 存储 和 树 型 存储 都 创建 一 个 
equals() 方 法 ， 但 是 hashCodeO 只 有 在 这 个 类 将 会 被 置 于 HashSet (这 是 有 可 能 的 ， 因 为 它 通常 
是 你 的 Set 实 现 的 首选 ) 或 者 LinkedHashSet 中 时 才 是 必需 的 。 但 是 ， 对 于 良好 的 编程 风格 而 言 ， 
你 应 该 在 覆盖 equals0 方 法 时 ， 总 是 同时 覆盖 hashCode0 方 法 。 

下 面 的 示例 演示 了 为 了 成 功 地 使 用 特定 的 Set 实 现 类 型 而 必须 定义 的 方法 : 


71: containers/TypesForSets. java 
// Methods necessary to put your own type in a Set. 
import java.utit.*; 


class SetType { 
int i; 
public SetType(int n) { i =n; } 
public boolean equals (Object o) { 
return o instanceof SetType && (i == ((SetType)o). i); 


} 
public String toString() { return Integer.toString(i); } 
} 


class HashType extends SetType { 
public HashType(int n) { super(n); } 
public int hashCode() { return i: } 
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} 


class TreeType extends SetType 
implements Comparable<TreeType> { 
public TreeType(int n) { super(n); } 
public int compareTo(TreeType arg) 
return (arg.i < i ? -1 : (arg.i 
$ 
} 
public class TypesForSets { 
static <T> Set<T> fill(Set<T> set, Class<T> type) { 
try { 
for(int i = @; i < 10; i++) 
set.add( 
type. getConstructor(int.class) .newInstance(i)); 
} catch(Exception e) { 
throw new RuntimeException(e) ; 
} 
return set; 
} 
static <T> void test(Set<T> set, Class<T> type) { 
fill(set, type); 
fill(set, type); // Try to add duplicates 
fill(set, type); 
System. out.printin(set) ; 
} 
public static void main(String[] args) { 
test (new HashSet<HashType>(), HashType.class); 
test (new LinkedHashSet<HashType>(), HashType.class); 
test(new TreeSet<TreeType>(), TreeType.class); 
// Things that don't work: 
test (new HashSet<SetType>(), SetType.class); 
test (new HashSet<TreeType>(), TreeType.class); 
test (new LinkedHashSet<SetType>(), SetType.class) ; 
test (new LinkedHashSet<TreeType>(), TreeType.class) ; 
try { 
test (new TreeSet<SetType>(), SetType.class); 
} catch(Exception e) { 
System.out.printIn(e.getMessage()) ; 
} 
try { 
test (new TreeSet<HashType>(), HashType.class); 
} catch(Exception e) { 
System. out.printin(e.getMessage()); 





} 
} 
} /* Output: (Sample) 
[2, 4, 9, 8, 6, 1, 3, 7, 5, 0] 
10, 1, 2,3. 4,5, 6,7, 8 9} 
19, 8,7, 6.5, 4, 3, 2, 1, 6) 
19,9, 7,5, 1,2, 63,0, 7,2, 4, 4 7.9 1, 3, 6, 2, 


4, 3, 0, 5S, 0, 8 8, 8, 6, 5, 1) 

(0, 5,5, 6, 5, 0, 3.1, 9, 8 4, 2,3,9. 7.3, 4, 4, 8, 
7,1, 9, 6, 2. 1, 8 2, 8 6, 7] 

{0. 1, 2, 3, 4, 5, 6, 7, 8. 9, 8, 1, 2,3, 4,5, 6, 7, 8 
9.0.1, 2, 3, 4, 5, 6, 7, 8, 9] 

(0, 1, 2, 3, 4, 5,6, 7, 8, 9, 8, 1, 2,3, 4, 5,6, 7, 8 
9,0, 1,2, 3. 4, 5. 6. 7, 8 9) 

java. lang.ClassCastexception: SetType cannot be cast to 
java. lang.Comparable 

java.lang.ClassCast€xception: HashType cannot be cast to 


java. lang. Comparable 
?1/ 


为 了 证 明 哪些 方法 对 于 某 种 特定 的 Set 是 必需 的 ， 并 且 同 时 还 要 避免 代码 重复 ， 我 们 创建 了 
个 类 。 基 类 SetType 只 存储 一 个 int， 并 且 通 过 toString0 方 法 产生 它 的 值 。 因 为 所 有 在 Set 中 存 
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储 的 类 都 必须 具有 equals( 方 法 ， 因 此 在 基 类 中 也 有 该 方 法 。 其 等 价 性 是 基于 这 个 int 类 型 的 i 的 
值 来 确定 的 。 

HashType 继 承 自 SetType， 并 且 添加 了 hashCode0 方 法 ， 该 方法 对 于 放置 到 Set 的 散 列 实现 
中 的 对 象 来 说 是 必需 的 。 

TreeType 实 现 了 Comparable 接 口 ， 如 果 一 个 对 象 被 用 于 任何 种 类 的 排序 容器 中 ， 例 如 
SortedSet (TreeSet 是 其 唯一 实现 ) ， 那 么 它 必须 实现 这 个 接口 。 注 意 ， 在 compareTo0 中 ， 我 没 
有 使 用 “简洁 明了 ”的 形式 return i- 记 ， 因 为 这 是 一 个 常见 的 编程 错误 ， 它 只 有 在 ji 和 li2 都 是 无 符 
号 的 int (如 果 Java 确 实 有 unsigned 关 键 字 的 话 ， 但 实际 上 并 没有 ) 时 才能 正确 工作 。 对 于 Java 的 
有 符号 int， 它 就 会 出 错 ， 因 为 int 不 够 大 ， 不 足以 表现 两 个 有 符号 int 的 差 。 例 如 i 是 很 大 的 正 整 
数 ， 而 j 是 很 大 的 负 整 数 ，i-j 就 会 溢出 并 且 返 回 负 值 ， 这 就 不 正确 了 。 

你 通常 会 希望 compareTo0 方 法 可 以 产生 与 equals0 方 法 一 致 的 自然 排序 。 如 果 equals0 对 于 
某 个 特定 比较 产生 true， 那 么 compareTo0 对 于 该 比较 应 该 返回 0， 如 果 equals0 对 于 某 个 比较 产 
生 false， 那 么 compareTo0 对 于 该 比较 应 该 返回 非 0 值 。 

在 TypesForSets 中 ，fill0 和 test0 方 法 都 是 用 泛 型 定义 的 ， 这 是 为 了 避免 代码 重复 。 为 了 验证 
某 个 Set 的 行为 ，test0 会 在 被 测 Set 上 调用 fl0 三 次 ， 尝 试 着 在 其 中 引入 重复 对 象 。fill0 方 法 可 以 
接受 任何 类 型 的 Set， 以 及 其 相同 类 型 Class 对 象 ， 它 使 用 Class 对 象 来 发 现 并 接受 int 参 数 的 构造 
器 ， 然 后 调用 该 构造 器 将 元 素 添加 到 Set 中 。 

从 输出 中 可 以 看 到 , HashSet 以 某 种 神秘 的 顺序 保存 所 有 的 元 素 (这 将 在 本 章 稍 后 进行 解释 )， 
LinkedHashSet 按 照 元 素 插 入 的 顺序 保存 元 素 ， 而 TreeSet 按 照排 序 顺序 维护 元 素 (按照 
compareTo0 的 实现 方式 ， 这 里 维护 的 是 降序 ) 。 

如 果 我 们 尝试 着 将 没有 恰当 地 支持 必需 的 操作 的 类 型 用 于 需要 这 些 方法 的 Set， 那 么 就 会 有 
大 麻烦 了 。 对 于 没有 重新 定义 hashCode( 方 法 的 SetType 或 TreeType， 如 果 将 它们 放置 到 任何 散 
列 实现 中 都 会 产生 重复 值 ， 这 样 就 违反 了 Set 的 基本 契约 。 这 相当 烦人 ， 因 为 这 甚至 不 会 有 运行 
时 错误 。 但 是 ， 默 认 的 hashCode0 是 合法 的 ， 因 此 这 是 合法 的 行为 ， 即 便 它 不 正确 。 确 保 这 种 程 
序 的 正确 性 的 唯一 可 靠 方法 就 是 将 单元 测试 合并 到 你 的 构建 系统 中 (请 查看 http://MindView. 
net/BooksBetterJava 处 的 补充 材料 以 了 结 更 多 的 信息 ) 。 

如 果 我 们 尝试 着 在 TreeSet 中 使 用 没有 实现 Comparable 的 类 型 ， 那 么 你 将 会 得 到 更 确定 的 结 
果 : 在 TreeSet 试 图 将 该 对 象 当 作 Comparable 使 用 时 ， 将 抛 出 一 个 异常。 

17.6.1 SortedSet 

SortedSet 中 的 元 素 可 以 保证 处 于 排序 状态 ， 这 使 得 它 可 以 通过 在 SortedSet 接 口中 的 下 列 方 
法 提供 附加 的 功能 : Comparator comparator0 返 回 当前 Set 使 用 的 Comparator， 或 者 返回 null， 
表示 以 自然 方式 排序 。 

Object first0 返回 容器 中 的 第 一 个 元 素 。 

Object last() 返回 容器 中 的 最 末 一 个 元 素 。 

SortedSet subSet(fromElement, toElement) 生成 此 Set 的 子 集 ， 范 围 从 fromElement (包含 ) 
到 toElement (不 包含 ) 。 

SortedSet headSet(toElement) 生成 此 Set 的 子 集 ， 由 小 于 toElement 的 元 素 组 成 。 

SortedSet tailSet(fromElement) 生成 此 Set 的 子 集 ， 由 大 于 或 等 于 fromElement 的 元 素 组 成 。 

下 面 是 一 个 简单 的 示例 : 


//: containers/SortedSetDemo. java 
// What you can do with a TreeSet. 
import java.util.*; 
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‘import static net.mindview.util.Print.*; 


public class SortedSetDemo { 
public static void main(String[] args) { 


SortedSet<String> sortedSet = new TreeSet<String>(); 


Collections. addAll(sortedSet, 
“one two three four five six seven eight” 
.Split(” ")): 
print(sortedSet) ; 
String low = sortedSet.first(); 
String high = sortedSet.last(); 
print (low) ; 
print (high); 
Iterator<String> it = sortedSet.iterator(); 
for(int i = 0; i <= 6; i++) { 
if(i == 3) low = it.next(); 
if(i == 6) high = it.next(); 
else it.next(); 
} 
print(low); 
print (high); 
print (sortedSet.subSet(low, high)); 
print (sortedSet .headSet(high)); 
print (sortedSet. tailSet (low)); 








} 
} 7* Output: 
{eight, five, four, one, seven, six, three, two) 
eight 
two 
one 
two 
lone, seven, six, three) 
(eight, five, four, one, seven, six, three] 
fone, seven, six, three, two] 
“Whim 


注意 ，SortedSet 的 意思 是 “ 按 对 象 的 比较 函数 对 元 素 排序 " ， 而 不 是 指 “ 元 素 插入 的 次 序 "。 


插入 顺序 可 以 用 LinkedHashSet 来 保存 。 


练习 9; (2) 使 用 RandomGenerator.String 来 填充 TreeSet， 但 是 要 使 用 字母 序 排序 。 打 印 这 


个 TreeSet， 并 验证 其 排序 顺序 。 


练习 10， (7) 使 用 LinkedList 作 为 底层 实现 ， 定 义 你 自己 的 SortedSet。 


17.7 队列 


除了 并 发 应 用 ，Queue 在 Java SE5 中 仅 有 的 两 个 实现 是 LinkedList 和 PriorityQueue， 它 们 的 
差异 在 于 排序 行为 而 不 是 性 能 。 下 面 是 涉及 Queue 实 现 的 大 部 分 操作 的 基本 示例 (并 非 所 有 的 
操作 在 本 例 中 都 能 工作 )， 包 括 基于 并 发 的 Quene。 你 可 以 将 元 素 从 队列 的 一 端 插 入 ， 并 于 另 一 


端 将 它们 抽取 出 来 : 


//: containers/QueueBehavior .java 

// Compares the behavior of some of the queues 
import java.util.concurrent.*; 

import java.util.*; 

‘import net.mindview.util.*; 


public class QueueBehavior { 
private static int count = 10; 


static <T> void test(Queue<T> queue, Generator<T> gen) { 


for(int i = @; i < count; i++) 

queue. offer (gen.next()); 
while(queue.peek() != null) 

System. out.print(queue.remove() + * "); 
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System.out.printin(); 
} 
static class Gen implements Generator<String> { 
String[] s = ("one two three four five six seven ”+ 
“eight nine ten").split(" "); 
int i: 
public String next() { return s[it+]: } 
} 
public static void main(String[] args) { 
test (new LinkedList<String>(), new Gen()): 
test (new PriorityQueue<String>(), new Gen()); 
test (new ArrayBlockingQueue<String>(count), new Gen()); 
test (new ConcurrentLinkedQueue<String>(), new Gen()); 
test (new LinkedBlockingQueue<String>(). new Gen()); 
test (new Prior ityBlockingQueue<String>(), new Gen()); 
} 
} /* Output: 
one two three four five six seven eight nine ten 
eight five four nine one seven six ten three two 
one two three four five six seven eight nine ten 
one two three four five six seven eight nine ten 
one two three four five six seven eight nine ten 
eight five four nine one seven six ten three two 
“i~ 


你 可 以 看 到 ， 除 了 优先 级 队列 ，Queue 将 精确 地 按照 元 素 被 置 于 Queue 中 的 顺序 产生 它们 。 


17.7.1 优先 级 队列 

在 第 11 章 曾经 给 出 过 优先 级 队列 的 一 个 简单 介绍 。 其 中 更 有 趣 的 问题 是 to-do 列 表 ， 该 列表 
中 每 个 对 象 都 包含 一 个 字符 串 和 一 个 主要 的 以 及 次 要 的 优先 级 值 。 该 列表 的 排序 顺序 也 是 通过 
实现 Comparable 而 进行 控制 的 : 


/1/1: containers/ToDoList. java 
// A more complex use of PriorityQueue. 
import java.util.*; 


class ToDoList extends PriorityQueue<ToDoList.ToDoItem> { 
static class ToDoItem implements Comparable<ToDoItem> { 
private char primary; 
private int secondary: 
private String item; 
public ToDoItem(String td, char pri, int sec) { 
primary = pri: 
secondary = sec; 
item = td; 
} 
public int compareTo(ToDoltem arg) { 
if (primary > arg.primary) 
return +1 
if (primary == arg.primary) 
if (secondary > arg.secondary) 
return +1; 
else if (secondary 
return 0; 
return -1; 
} 
public String toString() { 
return Character.toString(primary) + 
secondary + ": ”+ item; 








arg.secondary) 


} 

} * 

public void add(String td, char pri, int sec) { 
super.add(new ToDoItem(td, pri, sec)): 

} 

public static void main(String{] args) { 
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ToDoList toDoList = new ToDoList(); 
toDoList.add("Empty trash”, 'C', 4); 
toDoList.add("Feed dog", 'A', 2); 
toDoList.add("Feed bird: 
toDoList.add("Mow lawn" 
toDoList.add("Water lawi 
toDoList.add("Feed cat", 
while(!toDoList. isEmpty ()) 

System. out.printin(toDoList.remove()); 





} 

} /* Output: 
Al: Water lawn 
A2: Feed dog 
B1: Feed cat 
87: Feed bird 
C3: Mow lawn 
C4: Empty trash 
i~ 


你 可 以 看 到 各 个 项 的 排序 是 如 何 因为 使 用 了 优先 级 队列 而 得 以 自动 发 生 的 。 

练习 11: (2) 创建 一 个 类 ， 它 包含 一 个 Integer， 其 值 通过 使 用 java.util.Random 被 初始 化 为 0 
到 100 之 间 的 某 个 值 。 使 用 这 个 Integer 域 来 实现 Comparable。 用 这 个 类 的 对 象 来 填充 
PriorityQueue， 然 后 使 用 poll0 抽 取 这 些 值 以 展示 该 队列 将 按照 我 们 预期 的 顺序 产生 这 些 值 。 
17.7.2 双向 队列 

双向 队列 〈 双 端 队列 ) 就 像 是 一 个 队列 ， 但 是 你 可 以 在 任何 一 端 添加 或 移 除 元 素 。 在 
LinkedList 中 包含 支持 双向 队列 的 方法 ， 但 是 在 Java 标 准 类 库 中 没有 任何 显 式 的 用 于 双向 队列 的 
接口 。 ，LinkedList 无 法 去 实现 这 样 的 接口 ， 你 也 无 法 像 在 前 面 的 示例 中 转型 到 Queue 那 样 
去 向 上 转型 到 Deque。 但 是 ， 你 可 以 使 用 组 合 来 创建 一 个 Deque 类 ， 并 直接 从 LinkedList 中 峻 喜 
相关 的 方法 : 

//: net/mindview/util/Deque. java 

// Creating a Deque from a LinkedList. 


package net.mindview.util; 
import java.util.*; 





public class Deque<T> { 
private LinkedList<T> deque = new LinkedList<T>(); 
Public void addFirst(T e) { deque.addFirst(e); } 
Public void addLast(T e) { deque.addLast(e); } 
public T getFirst() { return deque.getFirst(): } 
public T getlast() { return deque.getLast(): } 
public T removeFirst() { return deque.removeFirst(); } 
public T removeLast() { return deque.removelast(); } 
public int size() { return deque.size(); } 
public String toString() { return deque.toString(); } 
// And other methods as necessary... 

} Mi~ 


如 果 将 这 个 Deque 用 于 自己 的 程序 中 ， 你 可 能 会 发 现 ， 为 了 使 它 实 用 ， 还 需要 增加 其 他 方法 。 
下 面 是 对 Deque 类 的 简单 测试 : 


//: containers/DequeTest. java 
import net.mindview.util.*; 
import static net.mindview.util.Print.*; 





public class DequeTest { 
static void fillTest(Deque<Integer> deque) { 
for(int i = 20; i < 27: i++) 
deque.addFirst(i); 
for(int 1 = 50: i < 55; i++) 
deque.addLast (i); 


容器 深入 研究 483 





public static void main(String[] args) { 

Deque<Integer> di = new Deque<Integer>(); 

filltest (di); 

print (di); 

while(di.size() != 9) 
printnb(di.removeFirst() + " "); 

print(); 

fillTest (di); 

while(di.sizeQ != 8) 
printnb(di.removeLast() +" "): 


} 
} /* Output: 
{26, 25, 24, 23, 22, 21, 20, 50, 51, 52, 53, 54) 
26 25 24 23 22 21 20 50 51 52 53 54 
54 53 52 51 50 20 21 22 23 24 25 26 
Whim 


你 不 太 可 能 在 两 端 都 放 入 元 素 并 抽取 它们 ， 因 此 ，Deque 不 如 Queue 那 样 常用 。 
17.8 理解 Map 


正如 你 在 第 11 章 中 所 学 到 的 ， 映射 表 (也 称 为 关联 数组 ) 的 基本 思想 是 它 维护 的 是 键 - 值 
(对 ) 关联 ， 因 此 你 可 以 使 用 键 来 查找 值 。 标 准 的 Java 类 库 中 包含 了 Map 的 几 种 基本 实现 ， 包 括 ; 
HashMap, TreeMap, LinkedHashMap, WeakHashMap, ConcurrentHashMap, 


IdentityHashMap。 它 们 都 有 同样 的 基本 接口 Map， 但 是 行为 特性 各 不 相同 ， 这 表现 在 效率 、 刍 
值 对 的 保存 及 呈现 次 序 、 对 象 的 保存 周期 、 映 射 表 如 何在 多 线程 程序 中 工作 和 判定 “ 键 ”等 价 
的 策略 等 方面 。Map 接 口 实现 的 数量 应 该 可 以 让 你 感觉 到 这 种 工具 的 重要 性 。 

你 可 以 获得 对 Map 更 深入 的 理解 ， 这 有 助 于 观察 关联 数组 是 如 何 创建 的 。 下 面 是 一 个 极其 
简单 的 实现 


1/: containers/AssociativeArray .java 
7) Associates keys with values. 
import static net.mindview.util.Print.*; 


public class AssociativeArray<K,V> { 
private Object{][] pairs; 
private int index; 
public AssociativeArray(int Length) { 
pairs = new Object{Length] [2] 


} 
public void put(K key, V value) { 
if (index >= pairs. length) 
throw new ArrayIndexOutOfBoundsException(); 
pairs[index++] = new Object[]{ key, value }: 


} 
@SuppressWarnings ("unchecked") 
public V get(K key) { 
for(int 1 = 0; i < index; i++) 
if (key .equals(pairs[i] [9])) 
return (V)patrs(i] [1]; 
return null; // Did not find key 


} 
public String toString() { 
StringBuilder result = new StringBuilder (): 
for(int i = 0; i < index; i++) { 
result.append(pairs{{] [9] .toString()): 
result.append(" : "); 
result. append(pairs{i] [1] .toString()): 
if(i < index - 1) 
result.append("\n"): 
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return result. toString(); 


} 
public static void main(String[] args) { 
AssociativeArray<String, String> map = 
new Associat iveArray<String, String> (6); 
map.put("sky", "blue"); 
map.put (“gra 
map.put ("oce 
map.put("tre 
map. put ("ear 
map.put("sun 
try { 
map.put("extra", "object"); // Past the end 
} catch(ArrayIndexOutOfBoundsException e) { 
print("Too many objects!"); 
} 
print(map) ; 
print (map.get("ocean")); 








} 
} /* Output: 
Too many objects! 
sky : blue 
grass : green 
ocean : dancing 
tree : tall 
earth : brown 
sun : warm 
dancing 
Wh 


关联 数组 中 的 基本 方法 是 put0 和 get0， 但 是 为 了 容易 显示 ，toString0 方 法 被 禾 盖 为 可 以 打 
印 键 - 值 对 。 为 了 展示 它 可 以 工作 ，mainO 用 字符 串 对 加 载 了 一 个 AssociativeArray， 并 打印 了 
所 产生 的 映射 表 ， 随 后 是 获取 一 个 值 的 get0。 

为 了 使 用 get(0 方 法 ， 你 需要 传递 想 要 查找 的 key， 然 后 它 会 将 与 之 相关 联 的 值 作为 结果 返回 ， 
或 者 在 找 不 到 的 情况 下 返回 null。 get0 方 法 使 用 的 可 能 是 能 想象 到 的 效率 最 差 的 方式 来 定位 值 的 : 
从 数组 的 头 部 开始 ， 使 用 equals0 方 法 依次 比较 键 。 但 这 里 的 关键 是 简单 性 而 不 是 效率 。 

因此 上 面 的 版 本 是 说 明 性 的 ， 但 是 缺乏 效率 ， 并 且 由 于 具有 固定 的 尺寸 而 显得 很 不 灵活 。 
幸运 的 是 ， 在 javautil 中 的 各 种 Map 都 没有 这 些 问题 ， 并 且 都 可 以 替代 到 上 面 的 示例 中 。 

练习 12， (1) 在 AssociativeArray.java 的 main() 中 替代 为 使 用 HashMap、TreeMap 和 
LinkedHashMap, 

练习 13: (4) 使 用 AssociativeArrayjava 来 创建 一 个 单词 出 现 次 数 的 计数 器 ， 用 String 映 射 到 
Integer。 使 用 本 书 中 的 net.mindview.util.TextFile 工 具 打开 一 个 文本 文件 ， 并 使 用 空格 和 标点 符 
号 将 该 文件 断 开 为 单词 ， 然 后 计数 该 文件 中 各 个 单词 出 现 的 次 数 。 

17.8.1 性 能 

性 能 是 映射 表 中 的 一 个 重要 问题 ， 当 在 getO 中 使 用 线性 搜索 时 ， 执 行 速度 会 相当 地 慢 ， 而 
这 正 是 HashMap 提 高 速度 的 地 方 。HashMap 使 用 了 特殊 的 值 ， 称 作 数 列 码 ， 来 取代 对 键 的 缓慢 
搜索 。 散 列 码 是 “相对 唯一 ”的 、 用 以 代表 对 象 的 int 值 ， 它 是 通过 将 该 对 象 的 某 些 信息 进行 转 
换 而 生成 的 。hashCodeO) 是 根 类 Object 中 的 方法 ， 因 此 所 有 Java 对 象 都 能 产生 散 列 码 。 


HashMap 就 是 使 用 对 象 的 hashCode0 进 行 快速 查询 的 ， 此 方法 能 够 显著 提高 性 能 。 


O 如 果 这 仍 不 能 满足 你 对 性 能 的 要 求 ， 那 么 你 还 可 以 通过 创建 自己 的 Map 来 进一步 提高 查询 速度 ， 并 且 令 新 的 
Map 只 针对 你 使 用 的 特定 类 型 ， 这 样 可 以 避免 与 Object 之 间 的 类 型 转换 操作 。 要 到 达 更 高 的 性 能 ， 速 度 狂 们 
可 以 参考 Donald Knuth 的 The Art of Computer Programming, Volume 3: Sorting and Searching, Second Edition 。 使 
用 数组 代替 溢出 桶 ， 这 有 两 个 好 处 : 第 一 ， 可 以 针对 磁盘 存储 方式 做 优化 ， 第 二 ， 在 创建 和 回收 单独 的 记录 
时 ， 能 节约 很 多 时 间 。 
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下 面 是 基本 的 Map 实 现 。 在 HashMap 上 打 星 号 表示 如 果 没 有 其 他 的 限制 ， 它 就 应 该 成 为 你 
的 默认 选择 ， 因 为 它 对 速度 进行 了 优化 。 其 他 实现 强调 了 其 他 的 特性 ， 因 此 都 不 如 HashMap 快 。 





‘HashMap* Map 基 于 散 列 表 的 实现 ( 它 取代 了 Hashtable)。 插 入 和 查询 “ 键 值 对 ”的 开销 是 固定 
的 。 可 以 通过 构造 器 设置 容量 和 负载 因子 ， 以 调整 容器 的 性 能 

LinkedHashMap 类 似 于 HashMap， 但 是 迭代 遍历 它 时 ， 取 得 “ 键 值 对 ”的 顺序 是 其 插入 次 序 ， 或 者 是 
最 近 最 少 使 用 (LRU) 的 次 序 。 只 比 HashMap 慢 一 点 ， 而 在 选 代 访问 时 反而 更 快 ， 因 为 
它 使 用 链表 维护 内 部 次 序 

TreeMap 基于 红 黑 树 的 实现 。 查 看 “ 键 ”或 “ 键 值 对 ”时 ， 它 们 会 被 排序 (次 序 由 Comparable 


或 Comparator 决 定 )。TreeMap 的 特点 在 于 ， 所 得 到 的 结果 是 经 过 排序 的 。TreeMap 是 唯 
一 的 带 有 subMap(0 方 法 的 Map， 它 可 以 返回 一 个 子 树 “… 


WeakHashMap HAE (weak key) 上 映射， 允许 释放 映射 所 指向 的 对 象 ， 这 是 为 解决 某 类 特殊 问题 而 设 
计 的 。 如 果 映 射 之 外 没有 引用 指向 某 个 “ 键 "， 则 此 “ 键 ”可 以 被 垃圾 收集 器 回收 

ConcurrentHashMap 一 种 线程 安全 的 Map， 它 不 涉及 同步 加 镇 。 我 们 将 在 “并 发 ”一 章 中 讨论 

IdentityHashMap 使 用 == 代 替 equals0 对 “ 键 ”进行 比较 的 散 列 映射 。 专 为 解决 特殊 问题 而 设计 的 





散 列 是 映射 中 存储 元 素 时 最 常用 的 方式 。 稍 后 你 将 会 了 解散 列 机 制 是 如 何 工作 的 。 

对 Map 中 使 用 的 键 的 要 求 与 对 Set 中 的 元 素 的 要 求 一 样 ， 在 TypesForSets.java 中 展示 了 这 一 
点 。 任 何 键 都 必须 具有 一 个 equals0 方 法 ， 如 果 键 被 用 于 散 列 Map， 那 么 它 必须 还 具有 恰当 的 
hashCode0 方 法 ， 如 果 键 被 用 于 TreeMap， 那 么 它 必须 实现 Comparable。 

下 面 的 示例 展示 了 通过 Map 接 口 可 用 的 操作 ， 这 里 使 用 了 前 面 定义 过 的 CountingMapData 
测试 数据 集 : 


//: containers/Maps.java 
// Things you can do with Maps. 

import java.util.concurrent.*; 

import java.util.*; 

import net.mindview.util.*; 

import static net.mindview.util.Print.*; 


public class Maps { 
public static voi 
printnb ("Siz 
printnb("Keys: "); 
print (map. reyset): /1 Produce a Set of the keys 





printKeys(Map<Integer String> map) { 
map.size() +". "); 





} 

public static void test(Map<Integer,String> map) { 
print(map. getClass().getSimpleName()); 
map.putAll (new Count ingMapData(25) 
// Map has ‘Set’ behavior for keys: 
map.putAll (new Count ingMapData(25)); 
printkeys(map) ; 
// Producing a Collection of the values: 
printnb("Values: "); 
print (map.values()); 
print (map) ; 
print ("m 











ntainsKey (11): ”+ map.containsKey(11));: 
print("map.get(11): " + map.get(11)); 
print("map.containsValue(\"FO\"): * 

+ map.containsVatue("FO")) ; 
Integer key = map.keySet().iterator().next(); 
print("First key in map: ”+ key); 
map. remove (key) ; 
printKeys (map) ; 
map.clear(); 
print("map.isEmpty(): ”+ map. isEmpty()); 
map.putAll (new Count ingMapData(25)); 
// Operations on the Set change the Map: 
map.keySet () removeAll (map.keySet()); 





























j= ~-—) <-> 
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print("map.isEmpty(): ”+ map.isEmpty()); 


} 

public static void main(String[] args) { 
test (new HashMap<Integer.String>()): 
test (new TreeMap<Integer,String>()): 
test (new LinkedHashMap<Integer ,String>()); 
test (new IdentityHashMap<Integer ,String>()); 
test (new ConcurrentHashMap<Integer ,String>()): 
test (new WeakHashMap<Integer ,String>()); 


} 
} /* Output: 
HashMap 
Size = 25, Keys: (15, 8, 23, 16, 7, 22, 9, 21, 6, 1, 14, 
24, 4, 19, 11, 18, 3, 12, 17, 2, 13, 20, 10, 5, 0J 
Values: [PO, 10, XO, Q0” HƏ, We, JO, VO, GƏ, BƏ, 08, YƏ, 

，56 Me, RO, Ne Ke, FO, AQ) 

» 22=WO, 9=J0, 21=V0, 6=G0, 
, 11=L@, 18=S0, 3=D0, 12=M0, 
+ 10=K0, 5=F0, @=A0} 








iaia rE true 

map.get(11): L@ 

map.containsValue("F0"): true 

First key in map: 15 

Size = 24, Keys: [8, 23, 16, 7, 22, 9, 21, 6, 1, 14, 24, 4, 

19, 11, 18, 3, 12, 17, 2, 13, 20, 10, 5, 0) 

map.isEmpty(): true 

map.isEmpty(): true 

ti~ 

printKeys0 展 示 了 如 何 生成 Map 的 Collection 视 图 。keySet0 方 法 返回 由 Map 的 键 组 成 的 Set。 
因为 在 Java SE5 提 供 了 改进 的 打印 支持 ， 你 可 以 直接 打印 values0 方 法 的 结果 ， 该 方法 会 产生 一 
个 包含 Map 中 所 有 “ 值 ”的 Collection。( 注 意 ， 键 必须 是 唯一 的 ， 而 值 可 以 有 重复 。) 由 于 这 些 
Collection 背 后 是 由 Map 支 持 的 ， 所 以 对 Collection 的 任何 改动 都 会 反映 到 与 之 相关 联 的 Map。 

此 程序 的 剩余 部 分 提供 了 每 种 Map 操 作 的 简单 示例 ， 并 测试 了 每 种 基本 类 型 的 Map。 

练习 14，(3) 说 明 java.util.Properties 在 上 面 的 程序 中 可 以 工作 。 

17.8.2 SortedMap 

使 用 SortedMap (TreeMap 是 其 现 阶段 的 唯一 实现 ) ， 可 以 确保 键 处 于 排序 状态 。 这 使 得 它 
具有 额外 的 功能 ， 这 些 功能 由 SortedqMap 接 口中 的 下 列 方法 提供 : 

Comparator comparator(); 返回 当前 Map 使 用 的 Comparator， 或 者 返回 null， 表 示 以 自然 
方式 排序 。T firstKey0 返 回 Map 中 的 第 一 个 键 。T lastKeyO 返 回 Map 中 的 最 末 一 个 键 。 
SortedMap subMap(fromKey, toKey) 生 成 此 Map 的 子 集 ， 范 围 由 fromKey (包含 ) 到 toKey (不 
包含 ) 的 键 确定 。SortedMap headMap(toKey) 生 成 此 Map 的 子 集 ， 由 键 小 于 toKey 的 所 有 键 值 
对 组 成 。SortedMap tailMap(fromKey) 生 成 此 Map 的 子 集 ， 由 键 大 于 或 等 于 fromKey 的 所 有 键 
值 对 组 成 。 

下 面 的 例子 与 SortedSetDemojava 相 似 ， 演 示 了 TreeMap 新 增 的 功能 : 


//: containers/SortedMapDemo. java 

// What you can do with a TreeMap. 

import java.util.*; 

import net.mindview.util.*; 

import static net.mindview.util.Print.*; 

` public class SortedMapDemo { 
public static void main(String[] args) { 
Treetap<Integer String> sortedMap = 
new TreeMap<Integer,String>(new Count ingMapData(10)): 

print (sortedMap) ; 
Integer low = sortedMap. firstKey(); 
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Integer high = sortedMap. lastKey(); 
Print(low) ; 
print (high); 
Iterator<Integer> it = sortedMap.keySet().iterator(); 
for(int i = @; i <= 6; i++) { 

if(i == 3) low = it.next(); 

if (i == 6) high = it.next(); 

else it.next(); 





print (low); 

print(high); 

print (sortedMap.subMap(low, high)); 
print (sortedMap.headMap(high)); 
print(sortedMap. tai lMap(low)) : 


} 
} /* Output: 
{O=AQ, 1=B0, 2=CO, 3=D0, 4=EO, 5=FO, 6=GƏ, 7=H9, 8=10, 






9=J0) 

日 

9 

3 

7 

{3=D0, 4=E0, 5=F0, 6=60} . 

{ =B0, 2=C0, 3=D0, . S=F0, 6=G0} 
{ 9, 5=F0, 6=60, . 8=10, 9=J0} 








此 处 ， 键 值 对 是 按键 的 次 序 排列 的 。TreeMap 中 的 次 序 是 有 意义 的 ， 因 此 “位 置 ”的 概念 才 有 
意义 ， 所 以 才能 取得 第 一 个 和 最 后 一 个 元 素 ， 并 且 可 以 提取 Map 的 子 集 。 
17.8.3 LinkedHashMap 

为 了 提高 速度 ，LinkedHashMap 散 列 化 所 有 的 元 素 ， 但 是 在 遍历 键 值 对 时 ， 却 又 以 元 素 的 
插入 顺序 返回 键 值 对 (System.outprintin0 会 迭代 人 遍历 该 映射 ， 因 此 可 以 看 到 人 遍历 的 结果 )。 此 
外 ， 可 以 在 构造 器 中 设 定 LinkedHashMap， 使 之 采用 基于 访问 的 最 近 最 少 使 用 (LRU) 算法 ， [838 
于 是 没有 被 访问 过 的 (可 被 看 作 需 要 删除 的 ) 元 素 就 会 出 现在 队列 的 前 面 。 对 于 需要 定期 清理 
元 素 以 节省 空间 的 程序 来 说 ， 此 功能 使 得 程序 很 容易 得 以 实现 。 下 面 就 是 一 个 简单 的 例子 ， 它 
演示 了 LinkedHashMap 的 这 两 种 特点 ， 


//: containers/LinkedHashMapDemo. java 
// What you can do with a LinkedHashMap. 
import java.util.*; 

import net.mindview.util.*; 

import static net.mindview.util.Print.*; 





public class LinkedHashMapDemo { 
public static void main(String(] args) { 
LinkedHashMap<Integer String> linkedMap = 
new LinkedHashMap<Integer, String>( 
new CountingMapData(9)); 
print (linkedMap) : 
// Least-recently-used order: 
linkedMap = 
new LinkedHashMap<Integer,String>(16, 0.75f, true); 
LinkedMap.putAl1(new Count ingMapData(9)); 
print(linkedMap) ; 
for(int 1 = 0; 1 < 6 
LinkedMap. get (1 
print (LinkedMa 
linkedMap. get (0: 
print(linkedMap. 
} 
} /* Output: 
{@=A0, 1=B0, 2=CO, 3=D0, 4=E9, 5=FO, 6=G9, 7=He, 8=10} 


; i++) // Cause accesses: 




















839) 
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=G8, 7=HO, 8=10} 
CO, 3=D8, 4=E0, 5=FO} 
k > 8. 3=D0, 4=E@, 5=F@, O=A0) 





在 输出 中 可 以 看 到 ， 键 值 对 是 以 插入 的 顺序 进行 遍历 的 ， 甚 至 LRU 算 法 的 版 本 也 是 如 此 。 
但 是 ， 在 LRU 版 本 中 ， 在 (只 ) 访问 过 前 面 六 个 元 素 后 ， 最 后 三 个 元 素 移 到 了 队列 前 面 。 然 后 
再 一 次 访问 元 素 “o” 时 ， 它 就 被 移 到 队列 后 端 了 。 


17.9 散 列 与 散 列 码 


在 第 11 章 的 例子 中 ， 标 准 类 库 中 的 类 被 用 作 HashMap 的 键 。 它 用 得 很 好 ， 因 为 它 具 备 了 键 
所 需 的 全 部 性 质 。 

当 你 自己 创建 用 作 HashMap 的 键 的 类 ， 有 可 能 会 忘记 在 其 中 放置 必需 的 方法 ， 而 这 是 通常 
会 犯 的 一 个 错误 。 例 如 ， 考 虑 一 个 天 气 预报 系统 ， 将 Groundhog (RE) 对 象 与 Prediction 
(预报 ) 对 象 联系 起 来 。 这 看 起 来 很 简单 。 创 建 这 两 个 类 ， 使 用 Groundhog 作 为 键 ，Prediction 
作为 值 : 


//: containers/Groundhog. java 
// Looks plausible, but doesn't work as a HashMap key. 


public class Groundhog { 
protected int number; 
public Groundhog(int n) { number = n; } 
public String toString() { 
return "Groundhog #” + number; 
$ 
) Mi~ 


//: containers/Prediction. java 
// Predicting the weather with groundhogs. 
import java.util.*; 


public class Prediction { 
private static Random rand = new Random(47); 
private boolean shadow = rand.nextDouble() > @.5; 
public String toString() { 
if (shadow) 
return "Six more weeks of Winter!"; 
else 
return "Early Spring!"; 


} 
} H~ 


//: containers/SpringDetector. java 

// What will the weather be? 

import java.lang.reflect.*; 

import java.util.*; 

import static net.mindview.util.Print.*; 


public class SpringDetector { 
// Uses a Groundhog or class derived from Groundhog: 
public static <T extends Groundhog> 
Void detectSpring(Class<T> type) throws Exception { 
Constructor<T> ghog = type.getConstructor(int.class); 
Map<Groundhog,Prediction> map = 
new HashMap<Groundhog, Prediction>(); 
for(int i = 0; i < 10; i++) 
map. put (ghog.newInstance(i), new Prediction()):; 
print("map = ”+ map); 
Groundhog gh = ghog.newInstance(3); 
print("Looking up prediction for ”+ gh); 
if (map. containsKey(gh)) 
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print(map.get(gh)); 
else 
print("Key not found: ”+ gh); 


public static void main(String[] args) throws Exception { 
detectSpring(Groundhog.class) ; 


} 

} /* Output: 

map = {Groundhog #3=Early Spring!, Groundhog #7=Early 
Spring!, Groundhog #5=Early Spring!, Groundhog #9=Six more 
weeks of Winter!, Groundhog #8=Six more weeks of Winter!, 
Groundhog #0=Six more weeks of Winter!, Groundhog #6=Early 
Spring!, Groundhog #4=S1x more weeks of Winter!, Groundhog 
#1=Six more weeks of Winter!, Groundhog #2=Early Spring!} 
Looking up prediction for Groundhog #3 

Key not found: Groundhog #3 

Wim 


每 个 Groundhog 被 给 予 一 个 标识 数字 ， 于 是 可 以 在 HashMap 中 这 样 查找 Prediction:“ 给 我 
与 Groundhog#3 相 关 的 Prediction。.”Prediction 类 包含 一 个 boolean 值 和 一 个 toString() 方 法 。 
boolean 值 使 用 java.utilrandom0 来 初始 化 ， 而 toString0 方 法 则 解释 结果 。detectSpring0 方 法 使 
用 反射 机 制 来 实例 化 及 使 用 Groundhog 类 或 任何 从 Groundhog 派 生出 来 的 类 。 如 果 我 们 为 解决 
当前 的 问题 从 Groundhog 继 承 创建 了 一 个 新 类 型 的 时 候 ，detectSpring0 方 法 使 用 的 这 个 技巧 就 
变 得 很 有 用 了 。 

首先 会 使 用 Groundhog 和 与 之 相关 联 的 Prediction 填 充 HashMap， 然 后 打印 此 HashMap， 
以 便 可 以 观察 它 是 否 被 填 人 了 一 些 内 容 。 然 后 使 用 标识 数字 为 3 的 Groundhog 作 为 键 ， 查 找 与 之 
对 应 的 预报 内 容 〈 可 以 看 到 ， 它 一 定 是 在 Map 中 ) 。 

这 看 起 来 够 简单 了 ， 但 是 它 不 工作 一 它 无 法 找到 数字 3 这 个 键 。 问 题 出 在 Groundhog 自 动 
地 继承 自 基 类 Object， 所 以 这 里 使 用 Object 的 hashCode( 方 法 生成 散 列 码 ， 而 它 默 认 是 使 用 对 象 
的 地 址 计算 散 列 码 。 因 此 ， 由 Groundhog(3) 生 成 的 第 一 个 实例 的 散 列 码 与 由 Groundhog(3) 生 成 
的 第 二 个 实例 的 散 列 码 是 不 同 的 ， 而 我 们 正 是 使 用 后 者 进行 查找 的 。 

可 能 你 会 认为 ， 只 需 编 写 恰当 的 hashCode0 方 法 的 覆盖 版 本 即 可 。 但 是 它 仍然 无 法 正常 运 
行 ， 除 非 你 同时 覆盖 equals0 方 法 ， 它 也 是 Object 的 一 部 分 。HashMap 使 用 equals0 判 断 当 前 的 
键 是 否 与 表 中 存在 的 键 相 同 。 

正确 的 equals0 方 法 必须 满足 下 列 5 个 条 件 : 

1) 自 反 性 。 对 任意 x，x.equals(x) 一 定 返 回 true。 

2) 对 称 性 。 对 任意 x 和 y， 如 果 y.equals(x) 返 回 true， 则 X.equals(y) 也 返回 true。 

3) 传递 性 。 对 任意 x、y、z， 如 果 有 x.equals(y) 返 回 ture，y.equals(z) 返 回 true， 则 x.equals(z) 
一 定 返 回 true。 

4) 一 致 性 。 对 任意 x 和 y， 如 果 对 象 中 用 于 等 价 比较 的 信息 没有 改变 ， 那 么 无 论调 用 
X.equals(y) 多 少 次 ， 返 回 的 结果 应 该 保持 一 致 ， 要 么 一 直 是 true， 要 么 一 直 是 false。 

5) 对 任何 不 是 null 的 x，x.equals(null) 一 定 返回 false。 

再 次 强调 ， 默 认 的 Object.equals0 只 是 比较 对 象 的 地 址 ， 所 以 一 个 Groundhog(3) 并 不 等 于 另 
一 个 Groundhog(3)。 因 此 ， 如 果 要 使 用 自己 的 类 作为 HashMap 的 键 ， 必 须 同时 重 载 hashCode0 
和 equals0， 如 下 所 示 : 


41: containers/Groundhog2. java 
// A class that's used as a key in a HashĦap 
// must override hashCode() and equals (). 


public class Groundhog2 extends Groundhog { 
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public Groundhog2(int n) { super(n); } 
public int hashCode() { return number; } 
public boolean equals (Object o) { 
return o instanceof Groundhog? && 
(number == ((Groundhog2)o). number); 


} 
} Mi~ 
//: containers/SpringDetector2. java 
// A working key. 


public class SpringDetector2 { 
public static void main(String{] args) throws Exception { 
SpringDetector .detectSpr ing (Groundhog2.class); 


} 
} /* Output: 
map = {Groundhog #2=Early Spring!, Groundhog #4=Six more 
weeks of Winter!, Groundhog #9=Six more weeks of Winter!, 
Groundhog #8=Six more weeks of Winter!, Groundhog #6=Early 
Spring!, Groundhog #1=Six more weeks of Winter!, Groundhog 
#3=Early Spring!, Groundhog #7=Early Spring!, Groundhog 
#5=Early Spring!, Groundhog #0=Six more weeks of Winter!) 
Looking up prediction for Groundhog #3 
Early Spring! 
Wh 


Groundhog2.hashCode0 返 回 Groundhog 的 标识 数字 (编号) 作为 散 列 码 。 在 此 例 中 ， 程 序 
员 负 责 确保 不 同 的 Groundhog 具 有 不 同 的 编号 。hashCode0 并 不 需要 总 是 能 够 返回 唯一 的 标识 
码 ( 稍 后 读者 会 理解 其 原因 ) ， 但 是 equals() 方 法 必须 严格 地 判断 两 个 对 象 是 否 相同 。 此 处 的 
equals0 是 判断 Groundhog 的 号 码 ， 所 以 作为 HashMap 中 的 键 ， 如 果 两 个 Groundhog2 对 象 具有 
相同 的 Groundhog 编 号 ， 程 序 就 出 错 了 。 

尽管 看 起 来 equals() 方 法 只 是 检查 其 参数 是 否 是 Groundhog2 的 实例 (使 用 第 14 章 中 介绍 过 
的 instanceof 关 键 字 ) ， 但 是 instanceof 悄 悄 地 检查 了 此 对 象 是 否 为 null， 因 为 如 果 instanceof 左 边 
的 参数 为 mull， 它 会 返回 false。 如 果 equals0 的 参数 不 为 null 且 类 型 正确 ， 则 基于 每 个 对 象 中 实际 
的 mumber 数 值 进行 比较 。 从 输出 中 可 以 看 到 ， 现 在 的 方式 是 正确 的 。 

当 在 HashSet 中 使 用 自己 的 类 作为 键 时 ， 必 须 注意 这 个 问题 。 

17.9.1 理解 hashCode() 

前 面 的 例子 只 是 正确 解决 问题 的 第 一 步 。 它 只 说 明 ， 如 果 不 为 你 的 键 覆盖 hashCode0 和 
equals0， 那 么 使 用 散 列 的 数据 结构 (HashSet，HashMap，LinkedHashSet 或 LinkedHashMap) 
就 无 法 正确 处 理 你 的 键 。 然 而 ， 要 很 好 地 解决 此 问题 ， 你 必须 了 解 这 些 数 据 结构 的 内 部 构造 。 

首先 ， 使 用 散 列 的 目的 在 于 : 想 要 使 用 一 个 对 象 来 查找 另 一 个 对 象 。 不 过 使 用 TreeMap 或 
者 你 自己 实现 的 Map 也 可 以 达到 此 目的 。 与 散 列 实 现 相 反 ， 下 面 的 示例 用 一 对 ArrayLists 实 现 了 
一 个 Map。 与 AssociativeArray.java 不 同 ， 这 其 中 包含 了 Map 接 口 的 完整 实现 ， 因 此 提供 了 
entrySet0 方 法 : 


//: containers/SlowMap. java 

// A Map implemented with ArrayLists. 
import java.util.*; 

import net.mindview.util.*; 


public class SlowMap<K,V> extends AbstractMap<K,V> { 
private List<K> keys = new ArrayList<K>(): 
private List<V> values = new ArrayList<V>(); 
public V put(K key, V value) { 
V oldValue = get(key): // The old value or null 
if (1keys.contains(key)) { 
keys.add(key) ; 





竹器 深入 研究 491 





values add (value) ; 
} else 

values. set (keys. indexOf (key), value); 
return oldvalue; 


} 
public V get(Object key) { // key is type Object, not K 
if (1keys. contains (key)) 
return null; 
return values.get (keys. indexOf(key)) ; 


} 

public Set<Map.Entry<K,V>> entrySet() { 
Set<Map.Entry<k,V>> set= new HashSet<Map. Entry<K,V>>(); 
Iterator<k> ki = keys.iterator(); 
Iterator<V> vi = values. iterator(); 
while (ki hasNext()) 

set.add(new MapEntry<K,V>(ki.next(), vi.next())); 

return set; 


} 

public static void main(String(] args) { 
SlowMap<String, String> m= new SlowMap<String, String> (); 
m.putAll (Countries .capitals(15)) ; 
System.out.printin(m); 
System. out.print1n(m. get ("BULGARIA")); 
System. out. printin(m.entrySet()); 


} 
} /* Output: 
{CAMEROON=Yaounde, CHAD=N'djamena, CONGO=Brazzaville, CAPE 
VERDE=Praia, ALGERIA=Aigiers, COMOROS=Moroni, CENTRAL 
AFRICAN REPUBLIC=Bangui, BOTSWANA=Gaberone, 
BURUNDI=Bujumbura, BENIN=Porto-Novo, BULGARIA=Sofia, < 
EGYPT=Cairo, ANGOLA=Luanda, BURKINA FASO=Quagadougou, 
DJIBOUTI=Di j ibouti} 
Sofia 
[CAMEROON=Yaounde, CHAD=N'djamena, CONGO=Brazzaville, CAPE 
VERDE=Praia, ALGERIA=Algiers, COMOROS=Moroni, CENTRAL 
AFRICAN REPUBLIC=Bangui, BOTSWANA=Gaberone, 
BURUNDI=Bujumbura, BENIN=Porto-Novo, BULGARIA=Sofia, 
EGYPT=Cairo, ANGOLA=Luanda, BURKINA FASO=Quagadougou, 
DJIBOUTI=D4j ibout1] 
Whim 


put0 方 法 只 是 将 键 与 值 放 入 相应 的 ArrayList。 为 了 与 Map 接 口 保持 一 致 ， 它 必须 返回 旧 的 
键 ， 或 者 在 没有 任何 旧 键 的 情况 下 返回 null。 

同样 遵循 了 Map 规 范 ，get0 会 在 键 不 在 SlowMap 中 的 时 候 产 生 null。 如 果 键 存在 ， 它 将 被 用 
来 查找 表示 它 在 keys 列 表 中 的 位 置 的 数值 型 索引 ， 并 且 这 个 数字 被 用 作 索 引 来 产生 与 values 列 表 
相关 联 的 值 。 注 意 ， 在 get0 中 key 的 类 型 是 Object， 而 不 是 你 所 期 望 的 参数 化 类 型 K (并 且 是 在 
AssociativeArray.java 中 真正 使 用 的 类 型 )。 这 是 将 泛 型 注入 到 Java 语 言 中 的 时 刻 如 此 之 晚 所 导致 
的 结果 一 一 如 果 泛 型 是 Java 语 言 最 初 就 具备 的 属性 ， 那 么 get0 就 可 以 执行 其 参数 的 类 型 

Map.entrySet0 方 法 必须 产生 一 个 Map.Entry 对 象 集 。 但 是 ，Map.Entry 是 一 个 接口 ， 用 来 
描述 依赖 于 实现 的 结构 ， 因 此 如 果 你 想 要 创建 自己 的 Map 类 型 ， 就 必须 同时 定义 Map.Entry 的 
实现 : 


//: containers/MapEntry. java 
7/ A simple Map.Entry for sample Map implementations. 
import java.util.*; 
public class MapEntry<K,V> implements Map.Entry<K,V> { 
private K key; 
private V value; 
public MapEntry(K key, V value) { 
this. key = key; 
this.value = value; 
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public K getKey() { return key; } 
public V getValue() { return value; } 
public V setValue(V v) { 
V result = value; 
value = v; 
return result; 
} 
public int hashCode() { 
return (key==null ? © : key.hashCode()) ^ 
(value==null ? @ : value.hashCode()); 


} 
Public boolean equals(Object o) { 
if(!(o instanceof MapEntry)) return false: 
MapEntry me = (MapEntry)o; 
return 
(key == null ? 
me.getKey() == null : key.equals(me.getKey())) && 
(value == null ? 
me.getValue()== null : value.equals(me.getValue())); 


} 
public String toString() { return key + "=" + value: } 
} Mi~ 


这 里 ， 这 个 被 称 为 MapEntry 的 十 分 简单 的 类 可 以 保存 和 读 取 键 与 值 ， 它 在 entrySet0 中 用 
来 产生 键 - 值 对 Set。 注 意 ，entrySet0 使 用 了 HashSet 来 保存 键 - 值 对 ， 并 且 MapEntry 采 用 了 一 种 
简单 的 方式 ， 即 只 使 用 key 的 hashCode0 方 法 。 尽 管 这 个 解决 方案 非常 简单 ， 并 且 看 起 来 在 
SlowMap.main0) 的 琐碎 测试 中 可 以 工作 ， 但 是 这 并 不 是 一 个 恰当 的 实现 ， 因 为 它 创建 了 键 和 值 
的 副本 。entrySet0 的 恰当 实现 应 该 在 Map 中 提供 视图 ， 而 不 是 副本 ， 并 且 这 个 视图 允许 对 原始 
映射 表 进 行 修改 (副本 就 不 行 )。 练 习 16 提 供 了 修正 这 个 问题 的 机 会 。 

注意 ， 在 MapEntry 中 的 equals() 方 法 必须 同时 检查 键 和 值 ， 而 hashCode() 方 法 的 含义 稍 
后 就 会 介绍 。SlowMap 的 内 容 的 String 表 示 是 由 在 AbstractMap 中 定义 的 toString0 方 法 自动 产 
生 的 。 

练习 15: (1) 使 用 SlowMap 重 复 练习 13。 

练习 16: (7) 将 Mapjava 中 的 测试 应 用 于 SlowMap， 验 证 并 修改 它 ， 使 其 能 正常 工作 。 

练习 17: (2) 令 SlowMap 实 现 完整 的 Map 接 口 。 

练习 18: (3) 参考 SlowMap.java， 创 建 一 个 SlowSet。 

17.9.2 为 速度 而 散 列 

SlowMapjava 说 明了 创建 一 种 新 的 Map 并 不 困难 。 但 是 正如 它 的 名 称 SlowMap 所 示 ， 它 不 
会 很 快 ， 所 以 如 果 有 更 好 的 选择 ， 就 应 该 放弃 它 。 它 的 问题 在 于 对 键 的 查询 ， 键 没有 按照 任何 
特定 顺序 保存 ， 所 以 只 能 使 用 简单 的 线性 查询 ， 而 线性 查询 是 最 慢 的 查询 方式 。 

散 列 的 价值 在 于 速度 : 散 列 使 得 查询 得 以 快速 进行 。 由 于 瓶颈 位 于 键 的 查询 速度 ， 因 此 解 
决 方案 之 一 就 是 保持 键 的 排序 状态 ， 然 后 使 用 Collections.binarySearch0O 进 行 查询 (有 一 个 练习 
会 带领 读者 走 完 这 个 过 程 )。 

散 列 则 更 进一步 ， 它 将 键 保存 在 某 处 ， 以 便 能 够 很 快 找到 。 存 储 一 组 元 素 最 快 的 数据 结构 
是 数组 ， 所 以 使 用 它 来 表示 键 的 信息 〈 请 小 心 留意 ， 我 是 说 键 的 信息 ， 而 不 是 键 本 身 )。 但 是 因 
为 数组 不 能 调整 容量 ， 因 此 就 有 一 个 问题 : 我 们 希望 在 Map 中 保存 数量 不 确定 的 值 ， 但 是 如 果 
键 的 数量 被 数组 的 容量 限制 了 ， 该 怎么 办 呢 ? 

答案 就 是 : 数组 并 不 保存 键 本 身 。 而 是 通过 键 对 象 生成 一 个 数字 ， 将 其 作为 数组 的 下 标 。 
这 个 数字 就 是 散 列 码 ， 由 定义 在 Object 中 的 、 且 可 能 由 你 的 类 覆盖 的 hashCode0 方 法 (在 计算 
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机 科学 的 术语 中 称 为 散 列 函数 ) 生成 。 

为 解决 数组 容量 被 固定 的 问题 ， 不 同 的 键 可 以 产生 相同 的 下 标 。 也 就 是 说 ， 可 能 会 有 冲突 。 
因此 ， 数 组 多 大 就 不 重要 了 ， 任 何 键 总 能 在 数组 中 找到 它 的 位 置 。 

于 是 查询 一 个 值 的 过 程 首 先 就 是 计算 散 列 码 ， 然 后 使 用 散 列 码 查询 数组 。 如 果 能 够 保证 没 
有 冲突 (如 果 值 的 数量 是 固定 的 ， 那 么 就 有 可 能 ) ， 那 可 就 有 了 一 个 完美 的 散 列 函数 ， 但 是 这 种 
情况 只 是 特例 ? 。 通 常 ， 冲 突 由 外 部 链接 处 理 : 数组 并 不 直接 保存 值 ， 而 是 保存 值 的 list。 然 后 
对 list 中 的 值 使 用 equals0 方 法 进行 线性 的 查询 。 这 部 分 的 查询 自然 会 比较 慢 ， 但 是 ， 如 果 散 列 函 
数 好 的 话 ， 数 组 的 每 个 位 置 就 只 有 较 少 的 值 。 因 此 ， 不 是 查询 整个 list， 而 是 快速 地 跳 到 数组 的 
某 个 位 置 ， 只 对 很 少 的 元 素 进行 比较 。 这 便 是 HashMap 会 如 此 快 的 原因 。 

理解 了 散 列 的 原理 ， 我 们 就 能 够 实现 一 个 简单 的 散 列 Map 了 : 


/1: containers/SimpleHashMap. java 
// A demonstration hashed Map. 
import java.util.*; 

import net.mindview.util.*; 


public class SimpleHashMap<K,V> extends AbstractMap<K,V> { 
// Choose a prime number for the hash table 
// size, to achieve a uniform distribution: 
static final int SIZE = 997; 
// You can't have a physical array of generics, 
// but you can upcast to one: 
@SuppressWarnings ("unchecked") 
LinkedList<MapEntry<K,V>>(] buckets = 
new LinkedL ist (SIZE) ; 
public V put(K key, V value) { 
V oldValue = null; 
int index = Math.abs(key.hashCode()) % SIZE; 
if (buckets[index] == null) 
buckets[index] = new LinkedList<MapEntry<K,V>>(); 
LinkedList<MapEntry<K,V>> bucket = buckets{ index]; 
MapEntry<K,V> pair = new MapEntry<K.V>(key, value); 
boolean found = false; 
ListIterator<MapEntry<k,V>> it = bucket.listIterator(); 
while(it.hasNext()) { 
MapEntry<K,V> iPair = it.next(); 
if (iPair.getKey().equals(key)) { 
oldValue = iPair.getValue(): 
it.set(pair); // Replace old with new 
found = true; 
break; 
? 


} 
if(!found) 
buckets [index] .add(pair); 
return oldValue; 
} 
public V get(Object key) 《 
int index = Math.abs(key.hashCode()) % SIZE; 
if (buckets(index] == null) return null; 
for (MapEntry<K,V> iPair : buckets(index]) 
if (iPair.getKey() .equals(key)) 
return iPair.getValue(); 
return null; 





} 
public Set<Map.Entry<K,V>> entryset() { 
Set<Map.Entry<K,V>> set= new HashSet<Map.Entry<K,V>>(); 


O 完美 的 散 列 函 数 在 Java SE5 的 EnumMap 和 EnumSet 中 得 到 了 实现 ， 因 为 emum 定 义 了 固定 数量 的 实例 。 请 查看 
第 19 章 。 
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for (LinkedList<MapEntry<K,V>> bucket : buckets) { 
if (bucket == null) continue; 
for (MapEntry<K,V> mpair : bucket) 
set.add(mpair) ; 


return set; 

} 

public static void main(String[] args) { 
SimpleHashMap<String, String> m = 

new SimpleHashMap<String,String>(); 

m.putAll (Countries.capitals(25)); 
System. out.printin(m) ; 
System.out.printin(m.get("ERITREA")); 
System. out.printIn(m.entrySet()); 


$ 
} /* Output: 
{CAMEROON=Yaounde, CONGO=Brazzaville, CHAD=N'djamena, COTE 
D'IVOIR (IVORY COAST)=Yamoussoukro, CENTRAL AFRICAN 
REPUBLIC=Bangui, GUINEA=Conakry, BOTSWANA=Gaberone, 
BISSAU=Bissau, EGYPT=Cairo, ANGOLA=Luanda, BURKINA 
FASO=Ouagadougou, ERITREA=Asmara, THE GAMBIA=Banjul, 
KENYA=Nairobi, GABON=Libreville, CAPE VERDE=Praia, 
ALGERIA=Algiers, COMOROS=Moroni, EQUATORIAL GUINEA=Malabo, 
BURUNDI=Bujumbura, BENIN=Porto-Novo, BULGARIA=Sofia, 
GHANA=Accra, DJIBOUTI=Dijibouti, ETHIOPIA=Addis Ababa} 
Asmara 
[CAMEROON=Yaounde, CONGO=Brazzaville, CHAD=N‘djamena, COTE 
D'IVOIR (IVORY COAST)=Yamoussoukro, CENTRAL AFRICAN 
REPUBLIC=Bangut, GUINEA=Conakry, BOTSWANA=Gaberone, 
BISSAU=Bissau, EGYPT=Cairo, ANGOLA=Luanda, BURKINA 
FASO=Ouagadougou, ERITREA=Asmara, THE GAMBIA=Banjul, 
KENYA=Nairobi, GABON=Libreville, CAPE VERDE=Praia, 
ALGERIA=Algiers, COMOROS=Moroni, EQUATORIAL GUINEA=Malabo, 
BURUNDI=Bujumbura, BENIN=Porto-Novo, BULGARIA=Sofia, 
GHANA=Accra, DJIBOUTI=D1jibouti, ETHIOPIA=Addis Ababa) 
Whim 


由 于 获 列 表 中 的 “ 档 位 ”(slot) 通常 称 为 柄 位 (bucket), LRA THA AT Pe BUA 
组 命名 为 bucket。 为 使 散 列 分 布 均匀 ， 桶 的 数量 通常 使 用 质数 e。 注 意 ,为 了 能 够 自动 处 理 冲突 ， 
使 用 了 一 个 LinkedList 的 数组 ， 每 一 个 新 的 元 素 只 是 直接 添加 到 list 末 尾 的 某 个 特定 桶 位 中 。 即 
使 Java 不 允许 你 创建 泛 型 数组 ， 那 你 也 可 以 创建 指向 这 种 数组 的 引用 。 这 里 ， 向 上 转型 为 这 种 
数组 是 很 方便 的 ， 这 样 可 以 防止 在 后 面 的 代码 中 进行 额外 的 转型 。 

对 于 put0 方 法 ，hashCode0 将 针对 键 而 被 调用 ， 并 且 其 结果 被 强制 转换 为 正 数 。 为 了 使 产 
生 的 数字 适合 bucket 数 组 的 大 小 ， 取 模 操作 符 将 按照 该 数组 的 尺寸 取 模 。 如 果 数 组 的 某 个 位 置 
是 null， 这 表示 还 没有 元 素 被 散 列 至 此 ， 所 以 ， 为 了 保存 刚 散 列 到 该 定位 的 对 象 ， 需 要 创建 一 个 
新 的 LinkedList。 一 般 的 过 程 是 ， 查 看 当前 位 置 的 list 中 是 否 有 相同 的 元 素 ， 如 果 有 ， 则 将 旧 的 
值 赋 给 oldValue， 然 后 用 新 的 值 取代 旧 的 值 。 标 记 found 用 来 跟踪 是 否 找到 (相同 的 ) 旧 的 键 值 
对 ， 如 果 没 有 ， 则 将 新 的 对 添加 到 list 的 末尾 。 

8et0 方 法 按照 与 pat0 方 法 相同 的 方式 计算 在 buckets 数 组 中 的 索引 (这 很 重要 ， 因 为 这 样 可 
以 保证 两 个 方法 可 以 计算 出 相同 的 位 置 ) 如 果 此 位 置 有 LinkedList 存 在 ， 就 对 其 进行 查询 。 

注意 ， 这 个 实现 并 不 意味 着 对 性 能 进行 了 调 优 ， 它 只 是 想 要 展示 散 列 映射 表 执行 的 各 种 操 


O 事实 证 明 ， 质 数 实际 上 并 不 是 散 列 三 的 理想 容量 。 近 来 ，( 经 过 广泛 的 测试 ) Java 的 散 列 函 数 都 使 用 2 的 整数 次 
方 。 对 现代 的 处 理 器 来 说 ， 除 法 与 求 剑 数 是 最 慢 的 操作 。 使 用 2 的 整数 次 方 长 度 的 散 列表 ， 可 用 掩 码 代替 除法 。 
因为 get0 是 使 用 最 多 的 操作 ， 求 余数 的 % 操 作 是 其 开销 最 大 的 部 分 ， 而 使 用 2 的 整数 次 方 可 以 消除 此 开销 (也 
可 能 对 hashCode0 有 些 影响 ) 。 
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作 。 如 果 你 浏览 一 下 java.util.HashMap 的 源 代码 ， 你 就 会 看 到 一 个 调 过 优 的 实现 。 同 样 ， 为 了 
简单 ，SimpleHashMap 使 用 了 与 SlowMap 相 同 的 方式 来 实现 entrySet0, 这 个 方法 有 些 过 于 简单 ， 
不 能 用 于 通用 的 Map。 

练习 19， (1) 使 用 SimpleHashMap 重 复 练习 13。 

练习 20: (3) 修改 SimpleHashMap， 令 其 能 够 报告 冲突 ， 并 添加 相同 的 数据 来 做 测试 ， 以 便 
能 够 看 到 冲突 。 

练习 21: (3) 修改 SimpleHashMap， 令 其 报告 要 探 询 多 少 次 才能 发 现 冲突 。 也 就 是 说 ， 插 入 
元 素 时 ， 对 Iterator 调 用 多 少 次 next0 才 能 在 LinkedList 中 发 现 此 元 素 已 经 存在 。 

练习 22: (4) 实现 SimpleHashMap 的 clear0 和 remove) 方法 。 

练习 23，(3) 令 SimpleHashMap 实 现 完整 的 Map 接 口 。 

练习 24: (5) 模仿 SimpleHashMap.java 中 的 例子 ， 写 一 个 SimpleHashSet， 并 做 测试 。 

练习 25，(6) 修改 MapEntry， 使 其 成 为 一 种 自 包含 的 单 向 链表 (每 个 MapEntry 应 该 都 有 一 
个 指向 下 一 个 MapEntry 的 前 向 链接 ) ， 从 而 不 用 对 每 个 桶 位 都 使 用 ListIterator。 修 改 Simple- 
HashMap.java 中 其 余 的 代码 ， 使 得 这 种 新 方式 可 以 正确 地 工作 。 

17.9.3 覆盖 hashCode() 

在 明白 了 如 何 散 列 之 后 ， 编 写 自己 的 hashCode0 就 更 有 意义 了 。 

首先 ， 你 无 法 控制 bucket 数 组 的 下 标 值 的 产生 。 这 个 值 依赖 于 具体 的 HashMap 对 象 的 容量 ， 
而 容量 的 改变 与 容器 的 充满 程度 和 负载 因子 (本章 稍 后 会 介绍 这 个 术语 ) 有 关 。hashCode0 生 
成 的 结果 ， 经 过 处 理 后 成 为 桶 位 的 下 标 (在 SimpleHashMap 中 ， 只 是 对 其 取 模 ， 模 数 为 bucket 
数组 的 大 小 ) 。 

设计 hashCode(0 时 最 重要 的 因素 就 是 : 无 论 何 时 ， 对 同一 个 对 象 调用 hashCode0 都 应 该 生 
成 同样 的 值 。 如 果 在 将 一 个 对 象 用 put0 添 加 进 HashMap 时 产生 一 个 hashCode0 值 ， 而 用 get0 取 
出 时 却 产生 了 另 一 个 hashCode0) 值 ， 那 么 就 无 法 重新 取得 该 对 象 了 。 所 以 ， 如 果 你 的 
hashCode() 方 法 依赖 于 对 象 中 易 变 的 数据 ， 用 户 就 要 当心 了 ， 因 为 此 数据 发 生变 化 时 ， 
hashCode0 就 会 生成 一 个 不 同 的 散 列 码 ， 相 当 于 产生 了 一 个 不 同 的 键 。 

此 外 ， 也 不 应 该 使 hashCode0 依 赖 于 具有 唯一 性 的 对 象 信息 ， 尤 其 是 使 用 this 的 值 ， 这 只 能 
产生 很 精 糕 的 hashCode0。 因 为 这 样 做 无 法 生成 一 个 新 的 键 ， 使 之 与 put0 中 原始 的 键 值 对 中 的 
键 相同 。 这 正 是 SpringDetectorjava 的 问题 所 在 ， 因 为 它 默认 的 hashCode0 使 用 的 是 对 象 的 地 址 。 
所 以 ， 应 该 使 用 对 象 内 有 意义 的 识别 信息 。 

下 面 以 String 类 为 例 。String 有 个 特点 ， 如果 程序 中 有 多 个 String 对 象 ， 都 包含 相同 的 字符 
串 序列 ， 那 么 这 些 String 对 象 都 映射 到 同一 块 内 存 区域 。 所 以 new String(“hello”) 生 成 的 两 个 
实例 ， 虽 然 是 相互 独立 的 ， 但 是 对 它们 使 用 hashCode0 应 该 生成 同样 的 结果 。 通 过 下 面 的 程序 
可 以 看 到 这 种 情况 : 


//: containers/StringHashCode. java 


public class StringHashCode { 
Public static void main(String[] args) { 
String(] hellos = "Hello Hello”.split(™ "); 
System. out. printIn(hellos [8] .hashCode( 
System. out .printin(hellos[1] .hashCode()); 





} 
} /* Output: (Sample) 
69609650 
69609650 
Whim 








851 














852 








A 





四 


496 ， HITE 





对 于 String 而 言 ，hashCode0 明 显 是 基于 String 的 内 容 的 。 

因此 ， 要 想 使 hashCode0 实 用 ， 它 必须 速度 快 ， 并 且 必须 有 意义 。 也 就 是 说 ， 它 必须 基于 
对 象 的 内 容 生成 散 列 码 。 记 得 吗 ， 散 列 码 不 必 是 独一无二 的 (应 该 更 关注 生成 速度 ， 而 不 是 唯 
一 性 )， 但 是 通过 hashCode0 和 equals0， 必 须 能 够 完全 确定 对 象 的 身份 。 

因为 在 生成 桶 的 下 标 前 ，hashCode0 还 需要 做 进一步 的 处 理 ， 所 以 散 列 码 的 生成 范围 并 不 
重要 ， 只 要 是 int 即 可 。 

还 有 另 一 个 影响 因素 : 好 的 hashCode0 应 该 产生 分 布 均匀 的 散 列 码 。 如 果 散 列 码 都 集中 
在 一 块 ， 那 么 HashMap 或 者 HashSet 在 某 些 区 域 的 负载 会 很 重 ， 这 样 就 不 如 分 布 均匀 的 散 列 
函数 快 。 

在 Effective Java Programming Language Guide (Addison-Wesley 2001) 这 本 书 中 ，Joshua Bloch 
为 怎样 写 出 一 份 像样 的 hashCode0 给 出 了 基本 的 指导 : 

1) 给 int 变 量 result 赋 予 其 个 非 零 值 常量 ， 例 如 17。 

D 为 对 象 内 每 个 有 意义 的 域 f ( 即 每 个 可 以 做 equals0 操 作 的 域 ) 计算 出 一 个 int 散 列 码 e; 





域 类 型 计 算 
boolean c=(f?0:1) 
byte, char, shortsžint c= (inbf 
long c= (inf ^ (f>>>32)) 
float c€ = FloatfloatTolntBits(f); 
double long I = Double.doubleToLongBits(f); 
c= (nd ^ (>>> 32)) 
Object, 其 equals0 调用 这 个 域 的 equals0) c=f.hashCodeO 
数组 对 每 个 元 素 应 用 上 述 规则 
3) 合并 计算 得 到 的 散 列 码 ; 
result = 37 * result + c; 


4) 返回 result。 
5) 检查 hashCode0 最 后 生成 的 结果 ， 确 保 相同 的 对 象 有 相同 的 散 列 码 。 
下 面 便 是 遵循 这 些 指导 的 一 个 例子 : 


//: containers/CountedString. java 

// Creating a good hashCode(). 

import java.util.*; 

import static net.mindview.util.Print.*; 


public class Countedstring { 
private static List<String> created = 
new ArrayList<String>() ; 
private String s; 
private int id = 0; 
public CountedString(String str) { 
s= str; 
created. add(s); 
// id is the total number of instances 
// of this string in use by CountedString: 
for(String s2 : created) 
if(s2.equals(s)) 
id++; 
} 
public String toString() { 
return "String: "+s +" id: "+ id + 
* hashCode(): ”+ hashCode(); 
$ 
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public int hashCode() { 
// The very simple approach: 
/1/ return s.hashCode() * id: 
// Using Joshua Bloch's recipe: 
int result = 17; 
result = 37 * result + s.hashCode(); 
result = 37 * result + id; 
return result; 


} 
public boolean equals(Object o) { 
return o instanceof CountedString && 
s.equals(((CountedString)o).s) && 
id == ((CountedString)o).id: 


} 
public static void main(String[] args) { 
Map<Countedstring,Integer> map = 
new HashMap<CountedStr ing, Integer>() ; 
CountedString{] cs = new CountedString(5]; 
for(int i = 0; 1 < cs.length: i++) (° 
cs[i] = new CountedString("hi"); 
map.put(cs{i], 1); // Autobox int -> Integer 


$ 

print (map) ; 

for(CountedString cstring : cs) { 
print("Looking up ”+ cstring); 
print(map.get(cstring)); 

} 


} 
} /* Output: (Sample) 
{String: hi id: 4 hashCode(): 146450=3, String: hi id: 1 
hashCode(): 146447=0, String: hi id: 3 hashCode(): 
146449=2, String: hi id: 5 hashCode(): 146451=4, String: hi 
id: 2 hashCode(): 146448=1) 
Looking up String: hi id: 1 hashCode(): 146447 
9 
Looking up String: hi id: 2 hashCode(): 146448 
1 
Looking up String: hi id: 3 hashCode(): 146449 
2 
Looking up String: hi id: 4 hashCode(): 146450 
3 
Looking up String: hi id: 5 hashCode(): 146451 
4 


Whim 


CountedString 由 一 个 String 和 一 个 id 组 成 ， 此 id 代表 包含 相同 String 的 CountedString 对 象 的 
编号 。 所 有 的 String 都 被 存储 在 static ArrayList 中 ， 在 构造 器 中 通过 和 迭代 人 遍历 此 ArrayList 完 成 对 


id 的 计算 。 


hashCode0 和 equals0 都 基于 CountedString 的 这 两 个 域 来 生成 结果 ， 如 果 它 们 只 基于 String 


或 者 只 基于 id， 不 同 的 对 象 就 可 能 产生 相同 的 值 。 


在 main0 中 ， 使 用 相同 的 String 创 建 了 多 个 CountedString 对 象 。 这 说 明 ， 虽 然 String 相 同 ， 
但 是 由 于 id 不 同 ， 所 以 使 得 它们 的 散 列 码 并 不 相同 。 在 程序 中 ，HashMap 被 打印 了 出 来 ， 因 此 
可 以 看 到 它 内 部 是 如 何 存储 元 素 的 〈 以 无 法 辨别 的 次 序 ) ， 然 后 单独 查询 每 一 个 键 ， 以 此 证 明 查 


询 机 制 工作 正常 。 


作为 第 二 个 示例 , 请 考虑 Individual 类 ， 它 被 用 作 第 14 章 中 所 定义 的 typeinfo.pet 类 库 的 基 类 。 
Individual 类 在 那 一 章 中 就 用 到 了 ， 而 它 的 定义 则 放 到 了 本 章 ， 因 此 你 可 以 正确 地 理解 其 实现 : 


//: typeinfo/pets/Individual .java 
package typeinfo.pets; 


public class Individual implements Comparable<Individual> { 
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private static long counter = 0; 
private final long id = counter++; 
private String name; 
public Individual (String name) { this.name = name; } 
// ‘name’ is optional: 
public Individuat() {} 
public String toString() { 
return getClass().getSimpleName() + 
(mame == null ? ** ; * * + name); 
} 
public long id() { return id; } 
public boolean equals(Object o) { 
return o instanceof Individual && 
id == ((Individual)o). id; 
} 
public int hashCode() { 
int result = 17; 
if(name != null) 
result = 37 * result + name.hashCode() ; 
result = 37 * result + (int)id; 
return result; 
} 
public int compareTo(Individual arg) { 
// Compare by class name first: 
String first = getClass().getSimpleName(); 
String argFirst = arg.getClass().getSimpleName(); 
int firstCompare = first.compareTo(argFirst): 
if(firstCompare != 6) 
return firstCompare; 
if(name != null && arg.name != null) { 
int secondCompare = name.compareTo(arg.name); 
if(secondCompare != 9) 
return secondCompare; 





} 
return (arg.id < id ? -1 : (arg.id == id ? @ : 1)); 


} 
} Hii 
compareTO( 方 法 有 一 个 比较 结构 ， 因 此 它 会 产生 一 个 排序 序列 ， 排 序 的 规则 首先 按照 实 
际 类 型 排序 ， 然 后 如 果 有 名 字 的 话 ， 按 照 mname 排 序 ， 最 后 按照 创建 的 顺序 排序 。 下 面 的 示例 说 
明了 它 是 如 何 工作 的 : 


//: containers/IndividualTest. java 
import holding.MapOfList; 

import typeinfo.pets 
import java.util.*; 





public class IndividualTest { 
public static void main(String{] args) { 
Set<Individual> pets = new TreeSet<Individual>(); 
for(List<? extends Pet> Ip : 
MapOfList.petPeople.values()) 
for(Pet p : Ip) 
pets.add(p); 
System. out.println(pets) ; 
} 
} /* Output: 
[Cat Elsie May, Cat Pinkola, Cat Shackleton, Cat Stanford 
aka Stinky el Negro, Cymric Molly, Dog Margrett, Mutt Spot, 
Pug Louie aka Louis Snorkelstein Dupree, Rat Fizzy, Rat 
Freckly, Rat Fuzzy) 
Wh 


由 于 所 有 的 宠物 都 有 名 字 ， 因 此 它们 首先 按照 类 型 排序 ， 然 后 在 同类 型 中 按照 名 字 排序 。 
为 新 类 编写 正确 的 hashCode0 和 equals0 是 很 需要 技巧 的 。Apache 的 Jakarta Commons 项 目 中 
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有 许多 工具 可 以 帮助 你 完成 此 事 ， 该 项 目 可 在 jakarta.apache.org/commons 的 lang 下 面 找 到 。( 此 项 
目 还 包括 许多 其 他 有 用 的 类 库 ， 而 且 它 似乎 是 Java 社 区 对 C++ 的 www.boost.org 作 出 的 回应 )。 

练习 26: (2) 在 CountedString 中 添加 一 个 char 域 ， 它 也 将 在 构造 器 中 初始 化 ， 然 后 修改 
hashCode0 和 equals() 方 法 ， 使 它们 都 包含 这 个 char 域 的 值 。 

练习 27: (3) 修改 CountedStringjava 中 的 hashCode0， 移 除 与 id 的 绑 定 ， 并 且 证 明 Counted- 
String 仍 能 正常 作为 键 使 用 。 这 种 方式 有 没有 问题 ? 

练习 28: (4) 修改 net/mindview/util/Tuple.java， 通 过 添加 hashCodeO 和 equals0 方 法 ， 并 为 
每 种 Tuple 类 型 都 实现 一 个 Comparable， 使 其 成 为 一 个 通用 类 。 


17.10 选择 接口 的 不 同 实现 


现在 已 经 知道 了 ， 尽 管 实际 上 只 有 四 种 容器 : Map、List、Set 和 Queue， 但 是 每 种 接口 都 
有 不 止 一 个 实现 版 本 。 如 果 需 要 使 用 某 种 接口 的 功能 ， 应 该 如 何 选择 使 用 哪 一 个 实现 呢 ? 

每 种 不 同 的 实现 各 自 的 特征 、 优 点 和 缺点 。 例 如 ， 从 容器 分 类 图 中 可 以 看 出 ，Hashtable、 
Vector 和 Stack 的 “特征 ”是 ， 它 们 是 过 去 遗留 下 来 的 类 ， 目 的 只 是 为 了 支持 老 的 程序 (最 好 不 
要 在 新 的 程序 中 使 用 它们 )。 

在 Java 类 库 中 不 同类 型 的 Queue 只 在 它们 接受 和 产生 数值 的 方式 上 有 所 差异 (你 将 在 第 21 章 
中 看 到 其 重要 性 ) 。 

容器 之 间 的 区 别 通常 归结 为 由 什么 在 背后 “支持 ”它们 。 也 就 是 说 ， 所 使 用 的 接口 是 由 什 
么 样 的 数据 结构 实现 的 。 例 如 ， 因 为 ArrayList 和 LinkedList 都 实现 了 List 接 口 ， 所 以 无 论 选择 哪 
一 个 ， 基 本 的 List 操 作 都 是 相同 的 。 然 而 ，ArrayList 底 层 由 数组 支持 ， 而 LinkedList 是 由 双向 链 
表 实现 的 ， 其 中 的 每 个 对 象 包含 数据 的 同时 还 包含 指向 链表 中 前 一 个 与 后 一 个 元 素 的 引用 。 因 
此 ， 如 果 要 经 常 在 表 中 插入 或 删除 元 素 ，LinkedList 就 比较 合适 (LinkedList 还 有 建立 在 
AbstractSequentialList 基 础 上 的 其 他 功能 ) ， 否 则 ， 应 该 使 用 速度 更 快 的 ArrayList。 

再 举 个 例子 ，Set 可 被 实现 为 TreeSet、HashSet 或 LinkedHashSet 9。 每 一 种 都 有 不 同 的 行为 : 
HashSet 是 最 常用 的 ， 查 询 速度 最 快 ，LinkedHashSet 保 持 元 素 插入 的 次 序 ，TreeSet 基 于 
TreeMap， 生 成 一 个 总 是 处 于 排序 状态 的 Set。 你 可 以 根据 所 需 的 行为 来 选择 不 同 的 接口 实现 。 

有 时 某 个 特定 容器 的 不 同 实现 会 拥有 一 些 共同 的 操作 ， 但 是 这 些 操作 的 性 能 却 并 不 相同 。 
在 这 种 情况 下 ， 你 可 以 基于 使 用 某 个 特定 操作 的 频率 ， 以 及 你 需要 的 执行 速度 来 在 它们 中 间 进 
行 选择 。 对 于 类 似 的 情况 ， 一 种 查看 容器 实现 之 间 差 异 的 方式 是 使 用 性 能 测试 。 

17.10.1 性 能 测试 框架 

为 了 防止 代码 重复 以 及 为 了 提供 测试 的 一 致 性 ， 我 将 测试 过 程 的 基本 功能 放置 到 了 一 个 测 
试 框架 中 。 下 面 的 代码 建立 了 一 个 基 类 ， 从 中 可 以 创建 一 个 匿名 内 部 类 列表 ， 其 中 每 个 匿名 内 
部 类 都 用 于 每 种 不 同 的 测试 ， 它 们 每 个 都 被 当 作 测 试 过 程 的 一 部 分 而 被 调用 。 这 种 方式 使 得 你 
可 以 很 方便 地 添加 或 移 除 新 的 测试 种 类 。 

这 是 模版 方法 设计 模式 的 另 一 个 示例 。 尽 管 你 遵循 了 典型 的 模版 方法 模式 ， 和 覆盖 了 每 个 特 
定 测试 的 Test.test0 方 法 ， 但 是 在 本 例 中 ， 其 核心 代码 (不 会 发 生变 化 ) 在 一 个 单独 的 Tester 类 
中 。。 待 测 容 器 类 型 是 泛 型 参数 C: 





O 或 者 实现 为 EnumSet 和 CopyOnWriteArraySet， 它 们 是 特例 。 尽 管 我 承认 各 种 不 同 的 容器 接口 都 可 能 拥有 额 
外 的 特殊 实现 ， 但 是 本 节 还 是 试图 只 浏览 那些 更 加 通用 的 情况 。 
© Krzysztof Sobolewski 才 助 我 设计 了 本 例 中 的 泛 型 。 
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//: containers/Test.java 
// Framework for performing timed tests of containers. 


public abstract class Test<C> { 
String name; 
public Test (String name) { this.name = name; } 
// Override this method for different tests. 
// Returns actual number of repetitions of test. 
abstract int test(C container, TestParam tp): 

} Mi~ 


每 个 Test 对 象 都 存储 了 该 测试 的 名 字 。 当 你 调用 test0 方 法 时 ， 必 须 给 出 待 测 容器 ， 以 及 
“信使 ”或 “数据 传输 对 象 " 它们 保存 有 用 于 该 特定 测试 的 各 种 参数 。 这 些 参 数 包括 size， 它 表 
示 在 容器 中 的 元 素数 量 ， 以 及 loops， 它 用 来 控制 该 测试 选 代 的 次 数 。 这 些 参数 在 每 个 测试 中 都 
有 可 能 会 用 到 ， 也 有 可 能 会 用 不 到 。 

每 个 容器 都 要 经 历 一 系列 对 test0 的 调用 ， 每 个 都 带 有 不 同 的 TestParam， 因 此 TestParam 还 
包含 静态 的 array0 方 法 ， 使 得 创建 TestParam 对 象 数组 变 得 更 容易 。array0 的 第 一 个 版 本 接受 的 
是 可 变 参数 列表 ， 其 中 包括 可 互 换 的 size 和 loops 的 值 ， 而 第 二 个 版 本 接受 相同 类 型 的 列表 ， 但 
是 它 的 值 都 在 String 中 一 一 通过 这 种 方式 ， 它 可 以 用 来 解析 命令 行 参数 ; 


//: containers/TestParam. java 
// A “data transfer object.” 


public class TestParam { 
public final int size; 
public final int loops; 
public TestParam(int size, int loops) { 
this.size = size; 
this.loops = loops; 


// Create an array of TestParam from a varargs sequence: 
public static TestParam[] array(int... values) { 

int size = values. length/2; 

TestParam[{] result = new TestParam{size]; 

int n = @; 

for(int 1 = 0; 1 < size; i++) 

result[i] = new TestParam(values[n++], values(n++]); 
return result; 


} 
// Convert a String array to a TestParam array: 
public static TestParam(] array(String[] values) { 
int[] vals = new int(values. length] ; 
for(int i = @; i < vals. length; i++) 
vals[i] = Integer.decode(values{1]); 
return array(vats) ; 


} 

} M~ 

为 了 使 用 这 个 框架 ， 你 需要 将 待 测 容器 以 及 Test 对 象 列表 传递 给 Tester.run0 方 法 (这 些 都 是 
重 载 的 泛 型 便利 方法 ， 它 们 可 以 减少 在 使 用 它们 时 所 必需 的 类 型 检查 )。Tester.run0 方 法 调用 适 
当 的 重 载 构造 器 ， 然 后 调用 timedTestO ， 它 会 执行 针对 该 容器 的 列表 中 的 每 一 个 测试 。 
timedTest0 会 使 用 paramList 中 的 每 个 TestParam 对 象 进行 重复 测试 。 因 为 paramList 是 从 静态 的 
defaultParams 数 组 中 初始 化 出 来 的 ， 因 此 你 可 以 通过 重新 赋值 defaultParams， 来 修改 用 于 所 有 
测试 的 paramList， 或 者 可 以 通过 传递 针对 菜 个 测试 的 定制 的 paramList， 来 修改 用 于 该 测试 的 
paramList; 

1/1: containers/Tester. java 


// Applies Test objects to lists of different containers. 
import java.util.*; 
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public class Tester<C> { 
public static int fieldWidth = 8: 
public static TestParam[] defaultParams= TestParam.array( 
10, 5000, 100, 5000, 1000, 5860, 10009, 509); 
// Override this to modify pre-test initialization: 
protected C initialize(int size) { return container; } 
protected C container; 
private String headline = ""; 
private List<Test<C>> tests; 
private static String stringField() { 
return "%" + fieldWidth + "s"; 2 
} 
private static String numberField() { 
return "%" + fieldWidth + "d"; 
} 
private static int sizeWidth = 
private static String sizeField = "%" + sizewidth + "s"; 
private TestParam(] paramList = defaultParams; 
public Tester(C container, List<Test<C>> tests) { 861 
this.container = container; 
this.tests = tests; 
if(container != null) 
headline = container. getClass().getSimpleName(): 
} 


public Tester(C container, List<Test<C>> tests, 
TestParam[] paramList) { 
this(container, tests); 
this.paramList = paramList; 





















} 
public void setHeadline(String newHeadline) { 
headline = newHeadline; 
l 
71 Generic methods for convenience : 
public static <C> void run(C cntnr, List<Test<C>> tests) { 
new Tester<C>(cntnr, tests).timedTest(); 
} 
public static <C> void run(C cntnr, 
List<Test<C>> tests, TestParam[] paramList) { 
new Tester<C>(cntnr, tests, paramList).timedTest(); 
} 
private void displayHeader() { 
// Calculate width and pad with '-': 
int width = fieldWidth * tests.size() + sizewidth; $ 
int dashLength = width - headline. length() - 1; 
StringBuilder head = new StringBuilder (width); 
for(int i = 9; i < dashLength/2; i++) 
head. append('-'); 
head.append(' ') 
head. append (headline) ; 
‘head. append(’ '); 
for(int i = 0; i < dashLength/2; i++) 
head. append('-'); 
System.out.printin(head) ; 
/1 Print column headers: 
System.out.format(sizeField, "size"): 
for(Test test : tests) 
System. out. format (stringField(), test.name); 
System. out.printin() ; 
} 
// Run the tests for this container: 
Public void timedTest() { 
displayHeader () ; 
i for(TestParam param : paramList) { 
System.out. format (sizeField, param. size); 
| for(Test<C> test : tests) { 
C kontainer = initialize(param.size); 
long start = System.nanoTime(); 
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// Call the overriden method: 
int reps = test.test(kontainer, param); 

long duration = System.nanoTime() - start; 

long timePerRep = duration / reps; // Nanoseconds 
System.out.format(numberField(), timePerRep); 


System.out.printin(); 


} 

} Mi~ 

stringField0 和 numberField0 方 法 会 产生 用 于 输出 结果 的 格式 化 字符 串 ， 格 式 化 的 标准 宽度 
可 以 通过 修改 静态 的 fieldWidth 的 值 进行 调整 。displayHeader(0 方 法 为 每 个 测试 格式 化 和 打印 头 
信息 。 

如 果 你 需要 执行 特殊 的 初始 化 ， 那 么 可 以 覆盖 initialize0 方 法 ， 这 将 产生 具有 恰当 尺寸 的 容 
器 对 象 一 一 你 可 以 修改 现 有 的 容器 对 象 ， 或 者 创建 新 的 容器 对 象 。 在 test0 方 法 中 可 以 看 到 ， 其 
结果 被 捕获 在 一 个 被 称 为 kontainer 的 局 部 引用 中 ， 这 使 得 你 可 以 将 所 存储 的 成 员 container 赫 换 
为 完全 不 同 的 被 初始 化 的 容器 。 

每 个 Test.test0 方 法 的 返回 值 都 必须 是 该 测试 执行 的 操作 的 数量 ， 这 些 测试 都 会 计算 其 所 有 
操作 所 需 的 纳 秒 数 。 你 应 该 意识 到 ， 通 常 System.nanoTime0 所 产生 的 值 的 粒度 都 会 大 于 1 (这 
个 粒度 会 随机 器 和 操作 系统 的 不 同 而 不 同 ) ， 因 此 ， 在 结果 中 可 能 会 存在 某 些 时 间 点 上 的 重合 。 

执行 的 结果 可 能 会 随机 器 的 不 同 而 不 同 ， 这 些 测试 只 是 想 要 比较 不 同 容器 的 性 能 。 
17.10.2 对 List 的 选择 

下 面 是 对 List 操 作 中 最 本 质 部 分 的 性 能 测试 。 为 了 进行 比较 ， 它 还 展示 了 Queue 中 最 重要 的 
操作 。 该 程序 创建 了 两 个 分 离 的 用 于 测试 每 一 种 容器 类 的 测试 列表 。 在 本 例 中 ，Queue 操 作 只 
应 用 到 了 LinkedList 之 上 。 


//: containers/ListPerformance. java 

// Demonstrates performance differences in Lists. 

// {Args: 108 500} Small to keep build testing short 
import java.util. *; 

import net.mindview.util.*; 


public class ListPerformance { 
static Random fand = new Random(); 
static int reps = 1000; 
Static List<Test<List<Integer>>> tests = 
new ArrayList<Test<List<Integer>>>(); 
static List<Test<LinkedList<Integer>>> qTests = 
new ArrayList<Test<LinkedList<Integer>>>(); 
static { 
tests.add(new Test<List<Integer>>("add") { 
int test(List<Integer> list, TestParam tp) { 
int loops = tp. loops; 
int listSize = tp.size; 
for(int i = 0; 1 < loops; i++) { 
list.clear( 
for(int j = 0: j < listSize; j++) 
List.add(j); 





$ 
return loops * listSize; 
} 
H: 
tests.add(new Test<List<Integer>>("get") { 
int test(List<Integer> list, TestParam tp) { 
int loops = tp.loops * reps; 
int listSiz ist.sizeQ); 
for(int i = 9; i < loops; i++) 
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List. get (rand. nextInt (ListSize)); 
return loops; 
} 
D: Pe 
tests.add(new Test<List<Integer>>("set") { 
int test(List<Integer> list, TestParam tp) { 
int loops = tp.loops * reps: 
int listSize = list.size(); 
for(int i = 9; i < loops; i++) 
List. set(rand.nextInt(listSize), 47); 
return loops; 
} 
ne 
tests.add(new Test<List<Integer>>("iteradd") { 
int test(List<Integer> list, TestParam tp) { 
final int LOOPS = 1000090; 
int half = list.size() 7 
ListIterator<Integer> it = list.listIterator(half); 
for(int 1 = 0; i < LOOPS; i++) 
it.add (47); 
return LOOPS; 
$ 
H; 
tests.add(new Test<List<Integer>>("insert") { 
int test(List<Integer> list, TestParam tp) { 
int loops = tp. loops; 
for(int i = 0; 1 < loops; i++) 
list.add(5, 47); // Minimize random-access cost 
return loops; 
} 
D; 
tests.add(new Test<List<Integer>>("remove") { 
int test(List<Integer> list, TestParam tp) { 
int loops = tp. loops; 
int size = tp.size: 
for(int i = 9; i < loops; i++) { 
list.clear(); 
Uist.addAll(new Count ingIntegerList(size)); 
while(list.size() > 5) 
List.remove(5); // Minimize random-access cost 









} 
return loops * size; 
} 
YD: 
// Tests for queue behavior: 
qTests.add(new Test<LinkedList<Integer>>("addFirst") { 
int test(LinkedList<Integer> list, TestParam tp) { 
int loops = tp. loops; 
int size = tp.size; 
for(int 1 = @; 1 < loops; i++) { 
list.clear(); 
for(int j = 0; j < size; j++) 
List.addFirst(47) ; 
} 
return loops * size; 
} 
Di 
qTests.add(new Test<LinkedList<Integer>>("addlast") { 
int test(LinkedList<Integer> list, TestParam tp) { 
int loops = tp. loops; 
int size = tp.size; 





for(int i = @; i < loops; i++) { 
list.clear(); 
for(int j = @; j < size; j++) 





list. addLast (47); 
} 


return loops * size; 
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} 
Yn: 
qTests.add( 
new Test<LinkedList<Integer>>("rmFirst") { 
int test (LinkedList<Integer> list, TestParam tp) { 
int loops = tp. loops; 
int size = tp.size; 
for(int i = 8; i < loops; i++) { 
list.clear(); 
list. addAll (new CountingIntegerList(size)); 
while(List.size() > 8) 
list. removeFirst(); 7 


+ 
return loops * size; 
} 
D: 
qTests.add(new Test<LinkedList<Integer>>("rmLast") { 
int test(LinkedList<Integer> list, TestParam tp) { 
int loops = tp. loops; 
int size = tp.size; 
for(int i = @; i < loops; i++) { 
list.clear(); 
List. addAll(new CountingIntegerList(size)); 
while(list.size() > 9) 
List. removeLast(); 
} 


return loops * size; 
} 
H: 





} 
static class ListTester extends Tester<List<Integer>> { 
public ListTester(List<Integer> container, 
List<Test<List<Integer>>> tests) { 
super(container, tests); 
} 
// Fill to the appropriate size before each test: 
@Override protected List<Integer> initialize(int size){ 
container.clear(); 
container .addAll (new Count ingIntegerList(size)): 
return container; 
} 
// Convenience method: 
public static void run(List<Integer> list, 
List<Test<List<Integer>>> tests) { 
new ListTester(list, tests).timedTest(); 
} 
} 
public static void main(String[] args) { 
if (args. length > 0) 
Tester.defaultParams = TestParam.array (args): 
// Can only do these two tests on an array: 
Tester<List<Integer>> arrayTest = 
new Tester<List<Integer>>(null, tests.subList(1, 3)){ 
// This Will be called before each test. It 
// produces a non-resizeable array-backed list: 
@0verride protected 
List<Integer> initialize(int size) { 
Integer[] ia = Generated. array(Integer.class, 
new CountingGenerator.Integer(), size); 
return Arrays.asList(ia); 
i } 
arrayTest.setHeadline("Array as List"); 
arrayTest.timedTest(); 
Tester .defauttParams= TestParam.array( 
10, 5000, 160, 5009, 1000, 1000, 16008, 200); 
if(args.length > 0) 
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Tester .defaultParams = TestParam. array (args); 
ListTester.run(new ArrayList<Integer>(), tests); 
ListTester.run(new LinkedList<Integer>(), tests); 
ListTester.run(new Vector<Integer>(), tests); 
Tester.fieldWidth = 12; 
Tester<LinkedList<Integer>> qTest = 

new Tester<LinkedList<Integer>>( 

new LinkedList<Integer>(), qTests); 
qTest.setHeadline("Queue tests"): 
qTest.timedTest(); 
} 
} /* Output: (Sample) 
--- Array as List --- 
size get set 







1000 129 165 

10000 129 165 

peeraa ArrayList --------------------- 
set iteradd insert remove 
191 435 446 






247 










set iteradd 
198 658 366 262 
230 457 108 201 
1000 133 1289 1353 430 136 239 
10000 172 13648 13187 435 255 239 


sate n eee eee tence tenes Vector ----------------------- 
size add get set iteradd insert remove 
10 129 145 187 299 3635 253 
100 72 144 198 263 3691 292 
1000 99 145 193 846 2162 927 
10000 108 145 186 6871 14730 7135 
i Queue tests -~------- 15 
size  addFirst addLast rmFirst rmLast 
10 199 163 251 253 
100 98 92 188 179 
1060 99 93 216 212 
10060 111 109 262 384 
Whi 


每 个 测试 都 需要 仔细 地 思考 ， 以 确保 可 以 产生 有 意义 的 结果 。 例 如 ，add 测 试 首先 清除 List， 
然后 重新 填充 它 到 指定 的 列表 尺寸 。 因 此 ， 对 clear0 的 调用 也 就 成 了 该 测试 的 一 部 分 ， 并 且 可 
能 会 对 执行 时 间 产 生 影 响 ， 尤 其 是 对 小 型 的 测试 。 尽 管 这 里 的 结果 看 起 来 相当 合理 ， 但 是 你 可 
以 设想 重 写 测试 框架 ， 使 得 在 计时 循环 之 外 有 一 个 对 准备 方法 的 调用 (在 本 例 中 将 包括 clear0 
调用 )。 

注意 ， 对 于 每 个 测试 ， 你 都 必须 准确 地 计算 将 要 发 生 的 操作 的 数量 以 及 从 test0 种 返回 的 值 ， 
因此 计时 是 正确 的 。 

get 和 set 测 试 都 使 用 了 随机 数 生成 器 来 执行 对 List 的 随机 访问 。 在 输出 中 ， 你 可 以 看 到 ， 对 
于 背后 有 数组 支撑 的 List 和 ArrayList， 无 论 列表 的 大 小 如 何 ， 这 些 访问 都 很 快速 和 一 致 ， 而 对 
于 LinkedList， 访 问 时 间 对 于 较 大 的 列表 将 明显 增加 。 很 显然 ,如 果 你 需要 执行 大 量 的 随机 访问 ， 
链接 链表 不 会 是 一 种 好 的 选择 。 

iteradd 测 试 使 用 迭代 器 在 列表 中 间 插入 新 的 元 素 。 对 于 ArrayList， 当 列表 变 大 时 ， 其 开销 
将 变 得 很 高 昂 ， 但 是 对 于 LinkedList， 相 对 来 说 比较 低廉 ， 并 且 不 随 列表 尺寸 而 发 生变 化 。 这 是 
因为 ArrayList 在 插入 时 ， 必 须 创建 空间 并 将 它 的 所 有 引用 向 前 移动 ， 这 会 随 ArrayList 的 尺寸 增 
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加 而 产生 高 昂 的 代价 。LinkedList 只 需 链接 新 的 元 素 ， 而 不 必修 改 列表 中 剩余 的 元 素 ， 因 此 你 可 
以 认为 无 论 列表 尺寸 如 何 变化 ， 其 代价 大 致 相同 。 

insert 和 remove 测 试 都 使 用 了 索引 位 置 5 作 为 插入 或 移 除 点 ， 而 没有 选择 List 两 端的 元 素 。 
LinkedList 对 List 的 端点 会 进行 特殊 处 理 一 这 使 得 在 将 LinkedList 用 作 Queue 时 ， 速 度 可 以 得 到 
提高 。 但 是 ， 如 果 你 在 列表 的 中 间 增加 或 移 除 元 素 ， 其 中 会 包含 随机 访问 的 代价 ， 我 们 已 经 看 
到 了 ， 这 在 不 同 的 List 实 现 中 变化 很 大 。 当 执行 在 位 置 5 的 插入 和 移 除 时， 随机 访问 的 代价 应 该 
可 以 被 忽略 , 但 是 我 们 将 看 不 到 对 LinkedList 端 点 所 做 的 任何 特殊 优化 操作 。 从 输出 中 可 以 看 出 ， 
在 LinkedList 中 的 插入 和 移 除 代价 相当 低廉 ， 并 且 不 随 列表 尺寸 发 生变 化 ， 但 是 对 于 ArrayList， 
插入 操 作 代价 特别 高 昂 ， 并 且 其 代价 将 随 列表 尺寸 的 增加 而 增加 。 

从 Queue 测 试 中 ， 你 可 以 看 到 LinkedList 可 以 多 么 快速 地 从 列表 的 端点 插入 和 移 除 元 素 ， 这 
正 是 对 Queue 行 为 所 做 的 优化 。 

通常 ， 你 可 以 只 调用 Tester'run0， 传 递 容器 和 tests 列 表 。 但 是 ， 在 这 里 我 们 必须 镍 盖 
initialize0 方 法 ， 使 得 List 在 每 次 测试 之 前 ， 都 会 被 清空 并 重新 填充 ， 否 则 在 不 同 的 测试 过 程 中 ， 
对 于 List 尺 寸 的 控制 将 丧失 。ListTester 继 承 自 Tester， 并 使 用 CountingIntegerList 执 行 这 种 初始 
化 。run0 便 捷 方法 也 被 覆盖 了 。 

我 们 还 想 将 数组 访问 与 容器 访问 进行 比较 (主要 是 与 ArrayList 比 较 )。 在 main0 的 第 一 个 测 
试 中 ， 使 用 匿名 内 部 类 创建 了 一 个 特殊 的 Test 对 象 。initialize0 方 法 被 禾 盖 为 在 每 次 被 调用 时 都 
创建 一 个 新 对 象 (此 时 会 忽略 container 对 象 ， 因 此 对 于 这 个 Tester 构 造 器 来 说 ，null 就 是 传递 进 
来 的 container 参 数 )。 这 个 新 对 象 是 使 用 Generated.array() 《这 是 在 第 16 章 中 定义 的 ) 和 
Arrays.asList0 创 建 的。 在 本 例 中 ， 只 有 两 个 测试 可 以 执行 ， 因 为 你 不 能 在 使 用 背后 有 数组 支撑 
的 List 时 ， 插 人 或 移 除 元 素 ， 因 此 ListsubList0 方 法 被 用 来 在 tests 列 表 中 选取 想 要 执行 的 测试 。 

对 于 随机 访问 的 get0 和 set0 操 作 ， 背 后 有 数组 支撑 的 List 只 比 ArrayList 稍 快 一 点 ， 但 是 对 于 
LinkedList， 相 同 的 操作 会 变 得 异常 引 人 注 目的 高 昂 ， 因 为 它 本 来 就 不 是 针对 随机 访问 操作 而 设 
计 的 。 

应 该 避免 使 用 Vector， 它 只 存在 于 支持 遗留 代码 的 类 库 中 (在 此 程序 中 它 能 正常 工作 的 唯 
一 原因 ， 只 是 因为 为 了 向 前 兼容 ， 它 被 适 配 成 了 List) 。 

最 佳 的 做 法 可 能 是 将 ArrayList 作 为 默认 首选 ， 只 有 你 需要 使 用 额外 的 功能 ， 或 者 当 程序 的 
性 能 因为 经 常 从 表 中 间 进 行 插入 和 删除 而 变 差 的 时 候 ， 才 去 选择 LinkedList。 如 果 使 用 的 是 固定 
数量 的 元 素 ， 那 么 既 可 以 选择 使 用 背后 有 数组 支撑 的 List (就 像 Arrays.asList0 产 生 的 列表 ) ， 也 
可 以 选择 真正 的 数组 。 

CopyOnWriteArrayList 是 List 的 一 个 特殊 实现 ， 专 门 用 于 并 发 编程 ， 我 们 将 在 第 21 章 中 讨 
论 它 。 

练习 29: (2) 修改 ListPerformance.java， 使 得 List 持 有 String 对 象 而 不 是 Integer。 使 用 第 16 
章 中 的 Generator 来 创建 测试 值 。 

练习 30，(3) 在 ArrayList 和 LinkedList 之 间 比 较 Collections.sort0 的 性 能 。 

练习 31: (5) 创建 一 个 封装 String 数 组 的 容器 ， 该 容器 只 允许 添加 和 读 取 String， 因 此 在 使 
用 过 程 中 不 存在 任何 转型 问题 。 如 果 在 添加 下 一 个 元 素 时 ， 内 部 数据 没有 足够 的 空间 ， 该 容器 
应 该 自动 调整 其 尺寸 。 在 main0 中 ， 比 较 你 的 容器 与 ArrayList<String> 的 性 能 。 

练习 32: (2) 重复 前 一 个 练习 ， 使 容器 中 包含 int， 并 与 与 ArrayList<String> 比 较 性 能 。 在 
性 能 比较 中 ， 包 括 递增 容器 中 每 个 对 象 的 处 理 。 

练习 33: (5) 创建 一 个 FastTraversalLinkedList， 为 了 快速 插入 与 移 除 ， 它 的 内 部 使 用 了 一 





= — 


BBRAFE 507 





个 LinkedList， 而 为 了 快速 遍历 和 getO 操 作 ， 则 使 用 了 一 个 ArrayList。 通 过 修改 ListPer- 
formance.java 来 测试 它 。 
17.10.3 微 基准 测试 的 危险 

在 编写 所 谓 的 微 基 准 测试 时 ， 你 必须 要 当心 ， 不 能 做 太 多 的 假设 ， 并 且 要 将 你 的 测试 空 化 ， 
以 使 得 它们 尽 可 能 地 只 在 感 兴趣 的 事项 上 花费 精力 。 你 还 必须 仔细 地 确保 你 的 测试 运行 足够 长 
的 时 间 ， 以 产生 有 意义 的 数据 ， 并 且 要 考虑 到 某 些 Java HotSpot 技 术 只 有 在 程序 运行 了 一 段 时 间 
之 后 才 会 踢 爆 问 题 (这 对 于 短期 运行 的 程序 来 说 也 很 重要 ) 。 

根据 计算 机 和 所 使 用 的 JVM 的 不 同 ， 所 产生 的 结果 也 会 有 所 不 同 ， 因 此 你 应 该 自己 运行 这 
些 测 试 ， 以 验证 得 到 的 结果 与 本 书 所 示 的 结果 是 否 相同 。 你 不 应 该 过 于 关心 这 些 绝对 数字 ， 将 
其 看 作 是 一 种 容器 类 型 与 另 一 种 之 间 的 性 能 比较 。 

剖析 器 可 以 把 性 能 分 析 工作 做 得 比 你 好 。Java 提 供 了 一 个 剖析 器 (查看 http://MinView.net/ 
Books/BetterJava 处 的 补充 材料 )， 另 外 还 有 很 多 第 三 方 的 自由 /开源 的 和 常用 的 剖析 器 可 用 。 

有 一 个 与 Math.random0O 相 关 的 示例 。 它 产生 的 是 0 到 1 的 值 吗 ? 包括 还 是 不 包括 1? 用 数学 
术语 表示 , 就 是 它 是 (0.1)、[0,1]、(0.1 还 是 [0,D)? ( 方 括号 表示 “包括 ”, 而 圆 括号 表示 “不 包括 ”。) 
下 面 的 测试 程序 也 许可 以 提供 答案 ; 


//; containers/RandomBounds .java 
// Does Math.random() produce 0.9 and 1.0? 
// {RunByHand} 

import static net.mindview.util.Print.*; 


public class RandomBounds { 
static void usage() ( 
print("Usage:"); 
print("\tRandomBounds lower"); 
print("\tRandomBounds upper"); 
System.exit(1); 


public static void main(String{] args) { 
if(args.length != 1) usage(); 
1f(args[6] .equals("lower")) { 
while(Math.random() != @.6) 
i // Keep trying 
print ("Produced 8.01"); 


} 

else if(args{0].equals("upper")) { 
while(Math.random() != 1.0) 

i // Keep trying 

print("Produced 1.0!"); 

} 

else 
usage(); 


} 

} Mi~ 

为 了 运行 这 个 程序 ， 你 需要 键入 下 面 两 行 命令 行 中 的 一 行 : 

java RandomBounds lower 
或 

java RandomBounds upper 

在 这 两 种 情况 中 ， 你 都 需要 手工 终止 程序 ， 因 此 看 起 来 好 像 Math.random0 永 远 都 不 会 产生 
0.0 或 1.0。 但 是 这 正 是 这 类 试验 产生 误导 之 所 在 。 如 果 你 选取 0 到 1 之 间 大 约 262 个 不 同 的 双 精 度 
小 数 ， 那 么 通过 试验 产生 其 中 任何 一 个 值 的 可 能 性 也 许 都 会 超过 计算 机 ， 甚 至 是 试验 员 本 身 的 
生命 周期 。 已 证 明 0.0 是 包含 在 Math.random0 的 输出 中 的 ， 按 照 数学 术语 ， 即 其 范围 是 [0.1)。 
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因此 ， 你 必须 仔细 分 析 你 的 试验 ， 并 理解 它们 的 局 限 性 。 
17.10.4 对 Set 的 选择 


可 以 根据 需要 选择 TreeSet、HashSet 或 者 LinkedHashSet。 下 面 的 测试 说 明了 它们 的 性 能 表 


现 ， 据 此 可 在 各 种 实现 间 做 出 权衡 选择 : 


//: containers/SetPerformance. java 

// Demonstrates performance differences in Sets. 

// {Args: 100 50009} Small to keep build testing short 
import java.util.*; 


public class SetPerformance { 
static List<Test<Set<Integer>>> tests = 
new ArrayList<Test<Set<Integer>>>(); 
static { 
tests.add(new Test<Set<Integer>> ("add") { 
int test(Set<Integer> set, TestParam tp) { 
int loops = tp. loops; 
int size = tp.size; 
for(int 1 = 0; i < loops; i++) { 
set.clear(); 
for(int j = @; j < size; j++) 
set.add(j); 
} 


return loops * size: 
} 
D: 
tests.add(new Test<Set<Integer>>("contains") { 
int test(Set<Integer> set, TestParam tp) { 
int loops = tp. loops; 
int span = tp.size * 
for(int 1 = 8; 1 < loops; i++) 
for(int j = @; j < span; j++) 
set.contains(j); 
return loops * span; 
} 
D: 
tests.add(new Test<Set<Integer>>("iterate") { 
int test(Set<Integer> set, TestParam tp) { 
int loops = tp.loops * 16; 
for(int 1 = @; i < loops: i++) { 
Iterator<Integer> it = set.iterator(); 
while (it.hasNext()) 
it.next(): 











} 
return loops * set.size(); 
} 
H: 
} 
public static void main(String{] args) { 
if (args.length > 0) 
Tester.defaultParams = TestParam.array(args); 
Tester. fieldwWidth = 10 
Tester.run(new TreeSet<Integer>(). tests); 
Tester.run(new HashSet<Integer>(), tests): 
Tester.run(new LinkedHashSet<Integer>(), tests): 
} 
} /* Output: (Sample) 





size add contains iterate 
10 746 173 89 
108 561 264 68 
1000 714 410 69 
10800 1975 552 69 
i HashSet --- 





size add contains iterate 
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10 308 91 
109 178 75 
1060 216 119 
10900 711 215 
--- LinkedHashSet -- 
size add contains 
10 350 65 
198 278 74 
1090 303 111 
10090 1615 256 


HashSet 的 性 能 基本 上 总 是 比 TreeSet 好 ， 特 别 是 在 添加 和 查询 元 素 时 ， 而 这 两 个 操作 也 是 
最 重要 的 操作 。TreeSet 存 在 的 唯一 原因 是 它 可 以 维持 元 素 的 排序 状态 ， 所 以 ， 只 有 当 和 需要 一 个 
排 好 序 的 Set 时 ， 才 应 该 使 用 TreeSet。 因 为 其 内 部 结构 支持 排序 ， 并 且 因为 欠 代 是 我 们 更 有 可 能 
执行 的 操作 ， 所 以 ， 用 TreeSet 选 代 通常 比 用 Hashset 要 快 。 

注意 ， 对 于 插入 操作 ，LinkedHashSet 比 HashSet 的 代价 更 高 ， 这 是 由 维护 链表 所 带 来 额外 
开销 造成 的 。 

练习 34: (1) 修改 SetPerformancejava， 使 Set 持 有 String 而 不 是 Integer 对 象 。 使 用 第 16 章 中 
的 Generator 来 创建 测试 值 。 


17.10.5 对 Map 的 选择 
下 面 的 程序 对 于 Map 的 不 同 实现 ， 在 性 能 开销 方面 给 出 了 指示 : 


//: containers/MapPerformance. java 

// Demonstrates performance differences in Maps. 

// {Args: 100 5000} Small to keep build testing short 
import java.util.*; 


public class MapPerformance { 
static List<Test<Map<Integer,Integer>>> tests = 
new ArrayList<Test<Map<Integer, Integer>>>(); 
static { 
tests.add(new Test<Map<Integer,Integer>>("put") { 
int test (Map<Integer ,Integer> map, TestParam tp) { 
int loops = tp. loops; 
int size = tp.size; 
for(int 1 = 0; f < loops; i++) { 
map.clear(); 
for(int j = @; j < size; j++) 
map.put(j, j): 





return loops * size; 
} 
D: 
tests.add(new Test<Map<Integer, Integer>>("get") { 
int test(Map<Integer ,Integer> map, TestParam tp) { 
int loops = tp. loops; 
int span = tp 
for(int i= @ 
for(int j = 
map. get (4); 
return loops * span: 





Di 
tests.add(new Test<Map<Integer, Integer>>("iterate") { 
int test(Map<Integer,Integer> map, TestParam tp) { 

int loops = tp.loops * 10; 
for(int i = @; i < loops; i ++) { 
Iterator it = map.entrySet().iterator(); 
while(it.hasNext()) 
it.next();* 
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return loops * map.size(); 


D: 

} 

public static void main(String[] args) { 
if (args. length > ©) 

Tester.defaultParams = TestParam.array(args) ; 
Tester.run(new TreeMap<Integer,Integer>(), tests); 
Tester.run(new HashMap<Integer,Integer>(), tests); 
Tester.run(new LinkedHashMap<Integer , Integer>(), tests); 
Tester. run( 

new IdentityHashMap<Integer,Integer>(), tests): 
Tester.run(new WeakHashMap<Integer, Integer>(), tests); 
Tester.run(new HashtablesInteger.Integer>(), tests); 





1000 385 222 





2787 341 
Ident ityHashMap 
put get iterate 
298 144 101 
204 287 132 K 
508 336 77 
767 266 56 
-- WeakHashMap -------- 
put get iterate 
484 146 151 X 
292 126 117 
411 136 152 
2165 138 555 
--------- Hashtable --------- 
size put get iterate 
10 264 113 113 
100 181 105 76 
10090 2690 201 80 
10000 1245 134 77 


H~ 3 

除了 IdentityHashMap， 所 有 的 Map 实 现 的 插入 操作 都 会 随 着 Map 尺 寸 的 变 大 而 明显 变 慢 。 
但 是 ， 查 找 的 代价 通常 比 插入 要 小 得 多 ， 这 是 个 好 消息 ， 因 为 我 们 执行 查找 元 素 的 操作 要 比 执 
行 插入 元 素 的 操作 多 得 多 。 À- 

Hashtable 的 性 能 大 体 上 与 HashMap 相 当 。 因 为 HashMap 是 用 来 替代 Hashtable 的 ， 因 此 它 
们 使 用 了 相同 的 底层 存储 和 查找 机 制 (你 稍 后 就 会 学 习 它 )， 这 并 没有 什么 令 人 吃惊 的 。 

TreeMap 通 常 比 HashMap 要 慢 。 与 使 用 TreeSet 一 样 ，TreeMap 是 一 种 创建 有 序列 表 的 方式 。 
树 的 行为 是 : 总 是 保证 有 序 ， 并 且 不 必 进 行 特殊 的 排序 。 一 旦 你 填充 了 一 个 TreeMap， 就 可 以 
调用 keySet0 方 法 来 获取 键 的 Set 视 图 ， 然 后 调用 toArray0 来 产生 由 这 些 键 构成 的 数组 。 之 后 ， 
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你 可 以 使 用 静态 方法 Arrays.binarySearch() 在 排序 数组 中 快速 查找 对 象 。 当 然 ， 这 只 有 在 
HashMap 的 行为 不 可 接受 的 情况 下 才 有 意义 ， 因 为 HashMap 本 身 就 被 设计 为 可 以 快速 查找 键 。 
你 还 可 以 很 方便 地 通过 单个 的 对 象 创建 操作 , 或 者 是 调用 putAll0， 从 TreeMap 中 创建 HashMap。 
最 后 ， 当 使 用 Map 时 ， 你 的 第 一 选择 应 该 是 HashMap， 只 有 在 你 要 求 Map 始 终 保持 有 序 时 ， 才 
需要 使 用 TreeMap。 

LinkedHashMap 在 插入 时 比 HashMap 慢 一 点 ， 因 为 它 维护 散 列 数据 结构 的 同时 还 要 维护 链 
表 (以 保持 插 人 顺序 )。 正 是 由 于 这 个 列表 ， 使 得 其 迭代 速度 更 快 。 

IdentityHashMap 则 具有 完全 不 同 的 性 能 ， 因 为 它 使 用 == 而 不 是 eqnals(0) 来 比较 元 素 。 
WeakHashMap 将 在 本 章 稍 后 介绍 。 

练习 35，(1) 修改 MapPerformance.java， 令 其 包含 对 SlowMap 的 测试 。 

练习 36，(5) 修改 SlowMap， 使 之 不 再 使 用 两 个 ArrayList， 而 是 只 持 有 一 个 以 MPair 对 象 组 
成 的 ArrayList。 验证 修改 后 的 版 本 是 否 工作 正常 。 使 用 MapPerformance.java 测 试 新 Map 的 速度 。 
然后 修改 put0 方 法 ， 令 其 插入 键 值 对 后 就 执行 sort0， 修 改 get0， 令 其 使 用 Collections.binary- 
Search0 查 询 健 。 比 较 新 旧版 本 的 性 能 。 

练习 37: (2) 修改 SimpleHashMap， 令 其 使 用 ArrayList 代 替 LinkedList。 修 改 MapPerformance 
java， 令 其 比较 两 种 不 同 实现 的 性 能 。 

HashMap 的 性 能 因子 

我 们 可 以 通过 手工 调整 HashMap 来 提高 其 性 能 ， 从 而 满足 我 们 特定 应 用 的 需求 。 为 了 在 调 
整 HashMap 时 让 你 理解 性 能 问题 ， 某 些 术语 是 必需 了 解 的 

eE: 表 中 的 桶 位 数 。 

“初始 容量 : 表 在 创建 时 所 拥有 的 桶 位 数 。HashMap 和 HashSet 都 具有 允许 你 指定 初始 容量 

的 构造 器 。 

RY: 表 中 当前 存储 的 项 数 。 

"负载 因子 : 尺寸 /容量 。 空 表 的 负载 因子 是 0， 而 半 满 表 的 负载 因子 是 0.5， 依 此 类 推 。 负 载 

轻 的 表 产 生 冲 突 的 可 能 性 小 ， 因 此 对 于 插入 和 查找 都 是 最 理想 的 〈 但 是 会 减 慢 使 用 迭代 器 

进行 遍历 的 过 程 )。HashMap 和 HashSet 都 具有 克 许 你 指定 负载 因子 的 构造 器 ， 表 示 当 负 

载 情况 达到 该 负载 因子 的 水 平时 ， 容 器 将 自动 增加 其 容量 〈 桶 位 数 ) ， 实 现 方式 是 使 容量 

大 致 加 倍 ， 并 重新 将 现 有 对 象 分 布 到 新 的 桶 位 集中 (这 被 称 为 再 散 列 ) 。 

HashMap 使 用 的 默认 负载 因子 是 0.75 (只 有 当 表 达到 四 分 之 三 满 时 ， 才 进行 再 散 列 ) ， 这 个 
因子 在 时 间 和 空间 代价 之 间 达 到 了 平衡 。 更 高 的 负载 因子 可 以 降低 表 所 需 的 空间 ， 但 是 会 增加 
查找 代价 ， 这 很 重要 ， 因 为 查找 是 我 们 在 大 多 数 时 间 里 所 做 的 操作 (包括 get0 和 put0)。 

如 果 你 知道 将 要 在 HashMap 中 存储 多 少 项 ， 那 么 创建 一 个 具有 恰当 大 小 的 初始 容量 将 可 以 
避免 自动 再 散 列 的 开销 。 。 

练习 38: (3) 在 JDK 文 档 中 查找 HashMap。 创 建 一 个 HashMap， 用 元 素 填充 它 ， 并 确定 其 


负载 因子 。 测 试 这 个 映射 表 的 查找 速度 ， 然 后 尝试 着 通过 创建 具有 更 大 的 初始 容量 的 新 的 


O 在 一 份 私 人 道 讯 中 ，Joshua Bloch 写 道 :“…… 我 相信 在 API 中 暴露 实现 细节 《例如 笋 列表 尺寸 和 负载 因子 ) 使 
我 们 误 入 歧途 。 客 户 端 应 用 可 以 告诉 我 们 集合 的 最 大 期 望 尺寸 ， 并 且 我 们 应 该 在 接口 中 接受 这 个 参数 。 但 是 ， 让 客户 端 
选择 这 些 参数 值 很 容易 变 得 弊 大 于 利 。 例 如 ， 考 虑 一 个 极端 的 例子 ，Vector 的 capacityincrement。 不 应 该 有 人 能 够 设置 
这 个 值 ， 我 们 也 不 应 该 提供 这 个 方法 。 如 果 将 这 个 值 设 置 为 任何 非 零 值 ， 那 么 在 序列 中 追加 空间 的 渐进 代价 将 从 线性 关 
系 变 为 二 次 关系 。 换 名 话说 ， 它 会 摧毁 程序 的 性 能 。 随 着 时 间 的 推移 ， 我 们 开始 渐渐 地 了 解 这 类 事情 。 如 果 你 看 看 
JdentityHashMap， 那 么 就 会 发 现 它 没有 任何 低级 别 的 调整 参数 .” 一 
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HashMap， 并 将 旧 映 射 表 中 的 元 素 复制 到 这 个 新 表 中 ， 来 创建 提高 查找 速度 ， 之 后 在 这 个 新 表 


上 再 次 运行 查找 速度 测试 程序 。 


练习 39: (6) 在 SimpleHashMap 中 添加 private rehash0 方 法 ， 它 将 在 负载 因子 超过 0.75 时 被 
调用 。 在 再 散 列 过 程 中 ， 先 求 出 桶 位 数量 加 倍 的 值 ， 然 后 搜索 大 于 这 个 值 的 第 一 个 质数 ， 将 其 


作为 新 的 桶 位 数 。 
17.11 实用 方法 


Java 中 有 大 量 用 于 容器 的 卓越 的 使 用 方法 ， 它 们 被 表示 为 java.util.Collections 类 内 部 的 静态 
方法 。 你 已 经 看 到 过 其 中 的 一 部 分 ， 例 如 addAIIO、reverseOrder0 和 binarySearchO。 下 面 是 另 
外 一 部 分 (synchronized 和 unmodifiable 的 使 用 方法 将 在 后 续 的 小 节 中 介绍 ) 。 在 这 张 表 中 ,在 


相关 的 情况 中 使 用 了 谤 型 ; 


checkedCollection(Collection<T>, Class<T> type) 
checkedList(List<T>, Class<T> type) 


checkedMap(Map<K,V>, Class<K> keyType, Class<V> valueType) 


checkedSet(Set<T>, Class<T> type) 


checkedSortedMap(SortedMap<K,V>, Class<K> keyType, 


Class<V> valueType) 
checkedSortedSet(SortedSet<T>, Class<T> type) 





产生 Collection 或 者 Collection 的 具体 子 类 型 的 
动态 类 型 安全 的 视图 。 在 不 可 能 使 用 静态 检查 
版 本 时 使 用 这 些 方法 

这 些 方法 在 第 15 章 中 的 “动态 类 型 安全 ” 标 
题 下 进行 过 说 明 





max(Collection) 
min(Collection) 


返回 参数 Collection 中 最 大 或 最 小 的 元 素 -采用 
Collection 内 置 的 自然 比较 法 





max(Collection, Comparator) 
min(Collection, Comparator) 


返回 参数 Colleetion 中 最 大 或 最 小 的 元 素 -- 采 
用 Comparator 进 行 比较 





indexOfSubList(List source, List target) 


返回 target 在 source 中 第 一 次 出 现 的 位 置 ， 或 
者 在 找 不 到 时 返回 -1。 





lastIndexOfSubList(List source, List target) 


返回 target 在 source 中 最 后 一 次 出 现 的 位 置 ， 
或 者 在 找 不 到 时 返回 -1 











replaceAll(List<T>, T oldVal, T newVal) 使 用 aewVal 符 换 所 有 的 oldVal 
reverse(List) RATRAT 
reverseOrder() 返回 一 个 Comparator， 它 可 以 道 转 实现 了 


reverseOrder(Comparator<T>) 


Comparator<T> 的 对 象 集合 的 自然 顺序 。 第 二 
个 版 本 可 以 逆转 所 提供 的 Comparator 的 顺序 





rotate(List, int distance) 


所 有 元 素 向 后 移动 distance 个 位 置 ， 将 末尾 的 
元 素 循环 到 前 面 来 











shuffle(List) 随机 改变 指定 列表 的 顺序 。 第 一 种 形式 提供 了 

shuffle(List, Random) 其 自己 的 随机 机 制 ， 你 可 以 通过 第 二 种 形式 提 
E 供 自己 的 随机 机 制 

sort(List<T>) 使 用 List<T> 中 的 自然 顺序 排序 。 第 二 种 形式 

sort(List<T>, Comparator<? Super T> c) 克 许 提供 用 于 排序 的 Comparator 

copy(List<? super T> dest, List<? extends T> sre) 将 src 中 的 元 素 复 制 到 dest 





swap(List, int i, int j) 


交换 list 中 位 置 i 与 位 置 j 的 元 素 。 通 常 比 你 自 
己 写 的 代码 快 
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( 续 ) 
fill(List<? super T>, T x) 用 对 象 x 替换 list 中 的 所 有 元 素 
nCopies(int n,T x) 返回 大 小 为 n 的 List<T>， 此 List 不 可 改变 ， 其 
中 的 引用 都 指向 x 
disjoint(Collection, Collection) 当 两 个 集合 没有 任何 相同 元 素 时 ， 返 回 true 
frequency(Collection, Object x) 返回 Collection 中 等 于 x 的 元 素 个 数 
emptyListO 返回 不 可 变 的 空 List、Map 或 Set。 这 些 方法 
emptyMap0 都 是 泛 型 的 ， 因 此 所 产生 的 结果 将 被 参数 化 为 
‘emptySet() 所 希望 的 类 型 
singleton(T x) 产生 不 可 变 的 Set<T>、List<T> 或 Map<K,V>， 
singletonList(T x) 它们 都 只 包含 基于 所 给 定 参数 的 内 容 而 形成 的 
singletonMap(K key, V value) 单一 项 
list(Enumeration <T> €) 产生 一 个 ArrayList<T> ， 它 包含 的 元 素 的 顺 
序 ， 与 (旧式 的 ) Enumeration (Iterator 的 前 
身 ) 返回 这 些 元 素 的 顺序 相同 。 用 来 转换 遗留 
的 老 代 码 
enumeration(Collection<T>) 为 参 





ec ar sid coe ees ee 
注意 ，min0 和 maxO 只 能 作用 于 Collection 对 象 ， 而 不 能 作用 于 List， 所 以 你 无 需 担 心 


Collection 是 否 应 该 被 排序 (如 前 所 述 ， 只 有 在 执行 binarySearch0 之 前 ， 才 确实 需要 对 List 或 数 
组 进行 排序 )。 


//: containers/Utilities.java 

// Simple demonstrations of the Collections utilities. 
import java.util.*; 

import static net.mindview.util.Print.*; 


public class Utilities ( 
static List<String> list = Arrays.asList( 
"one Two three Four five six one*.split(™ *)); 
public static void main(String[] args) { 
print(list); 
print("'list' disjoint (Four)?: " + 
Collections.disjoint(list, 
Collections. singletonList("Four"))); 
print("max: ”+ Collections.max(list)); 
print("min: * + Collections.min(list)); 
print("max w/ comparator: ”+ Collections.max(list, 
String. CASE_INSENSITIVE_ORDER)) ; 
print("min w/ comparator: ”+ Collections.min(list, 
String .CASE_INSENSITIVE_ORDER)) ; 
List<String> sublist = 
Arrays.asList("Four five six".split(" ")); 
print("indexOfSubList: * + 
Collections. indexOfSubList (list, sublist)); 
print("lastIndexOfSubList: ”+ 
Collections. lastIndexOfSubList (list, sublist)); 
Collections.replaceAll(list, "one", "Yo"); 
print(*replaceAll: " + list); 
Collections. reverse(list); 
print("reverse: " + list); 
Collections.rotate(list, 3); 
print(*rotate: " + list); 
List<String> source = 
Arrays.asList("in the matrix". split(" ")); 
Collections.copy(list, source): 
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print("copy: " + list); 
Collections.swap(list, @, list.size() ~ 1); 
print("swap: " + list); 
532 Collections. shuffle(list, new Random(47)); 2 
print("shuffled: "+ list); . 
Collections.fill(list, “pop"): 
print(*fill: ”+ list); 
print("frequency of ‘pop’: "+ 
Collections. frequency(list, “pop")); 
List<String> dups = Collections.nCopies(3, "snap"); 
print("dups: ”+ dups 
print("'list’ disjoint 'dups'?: * + 
Collections.disjoint(list, dups)); 
// Getting an old-style Enumeration: 
Enumeration<String> e = Collections.enumeration(dups) ; 
Vector<String> v = new Vector<String>(); wae 
while(e.hasMoreElements()) 
v.addElement (e.nextElement()); 
// Converting an old-style Vector 
// to a List via an Enumeration: 
ArrayList<String> arrayList = 
Collections. List(v.elements() 
print("arrayList: " + arrayList); 
} 
} /* Output: 
lone, Two, three, Four, five, six, one] 
‘List! disjoint (Four)?: false 
max: three 
min: Four 
max w/ comparator: Two 
min w/ comparator: five 
indexofSubList: 3 
lastIndexOfSubList: 3 
replaceAll: [Yo, Two, three, Four, five, six, Yo} 
reverse: [Yo, six, five, Four, three, Two, Yo) 
rotate: [three, Two, Yo, Yo, six, five, ‘Four] 
copy: [in, the, matrix, Yo, six, five, Four) 
swap: [Four, the, matrix, Yo, six, five, in] 
shuffled: [six, matrix, the, Four, Yo, five, in] 
fill: {pop, pop, pop, pop, pop, pop, pop) 
frequency of ‘pop’: 7 
dups: [snap, snap, snap) 
‘list’ disjoint ‘dups'?: true 
arrayList: [snap, snap, snap) 
Whim 


该 程序 的 输出 可 看 作 是 对 每 个 实用 方法 的 行为 的 解释 。 请 注意 由 于 大 小 写 的 缘故 而 造成 的 
使 用 String.CASE_INSENSITIVE_ORDER Comparator 时 min0 和 max0 的 差异 。 
17.11.1 List 的 排序 和 查询 

List 排 序 与 查询 所 使 用 的 方法 与 对 象 数组 所 使 用 的 相应 方法 有 相同 的 名 字 与 语法 ， 只 是 用 
Colleetions 的 static 方 法 代替 Arrays 的 方法 而 已 。 下 面 是 一 个 例子 ， 用 到 了 Utilitiesjava 中 的 list 
数据 : 


//: containers/ListSortSearch. java 

// Sorting and searching Lists with Collections utilities. 
import java.util.* 
import static net.mindview.util.Print.*; 





























public class ListSortSearch { 
public static void main(String{] args) { 
List<String> list = 
new ArrayList<String> (Utilities. list); 
List .addAll (Utilities list); 
print(list); 
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itrremove():; 
































t 
Leth me L ELUTE): 
Cotlections-sort(list, String~ CASEAINSENSTTIVESORDERT? 
print" Case- insensitive sorted: EOST); 
key = list.get(7)7 
indez = Collections. binarySearch list; key. 
String>CASE_INSENSITIVE—ORDER)> 要 
print("Location ef—" + key-+ " is " +-index + 
a ist Bet( + index+ "J =" + List.get(index)); 让 






































} 

} /# Oytput: 

[ones Two, three, Four, five, six, one, one, Two, threes 
Four? fiver .Six, one) 
Shuffledy (Four. tive, one, one: “Twos six, six; three, 
five, Four; Two: one, one] 

five,. one,. one; Two, sdx, six, three; 













ive, one, one, six, six; three, - 


Location “of «six-is 7; list-get(7) = six 

Casé-insefsitive sorted: ttive, five, Four, “one; one; six, 
six; threè; three; 
Location of three 
Whim 


ee PEs. 知 果 使 用 Comparator 进 行 排序 ， 那 么 binarySearch0 必 须 2 “ 
目 相 同 的 Comparator。 使 
HABE T CoMecaERGRUMMED sv RT 乱 List 的 顺序 : TEE 
AZE HEERE EB At 99 FEF EB A Ai E ANRE tH, 
练习 40; (5) MAT OSH String RAK F 并 使 其 成 为 Camparaple,- 因此 ;; 它们 之 间 
较 只 关 犀 第 一 个 Stritg。 通 过 使 用 &andomGeneratar 生 成 器 ， 用 这 个 类 的 对 象 填充 一 个 数 
E 明 排序 仍旧 可 以 正确 工作 ,使 用 你 的 Comparator 执 行 二 分 查找 。 
练习 41，(3 太 修改 前 一 个 练习 中 的 类 ， 使 其 可 以 作用 于 HashSet; 全 人 FaskMap 中 





] wt 
. list.get(7) = three | . < 






















it. 一 = 的 
练习 42: -(2) 修改 练习 40， 使 其 使 用 字母 序 排序 。 
1.2 设 定 Collection 或 Map 为 不 可 修改 17 


创建 一 个 只 读 的 Collection 或 Map， 有 时 可 以 带 来 某 些 方便 。Collections 类 可 以 帮助 达成 此 5 
9, CHT, SHARAMAS, BARABARA. DEAK ER, x 目 
Collection (Ai RRA HET Collection tL HE RAR). List, Set 和 Maps 下 例 将 说 明 如 ”885 应 
E 确 生成 各 种 只 读 容器 : | fa. 


//: containers/ReadOnly. java | 
// Using the Collections.unmodifiable methods 
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import java.util. *; 
import net.mindview.util.*: 
import static net.mindview.util.Print.*; 


public class ReadOnly { 
static Collection<String> data = 
new ArrayList<String>(Countries.names(6)): 
public static void main(String[] args) { 
Collection<String> c = 
Collections .unmodifiableCol lect ion( 
new ArrayList<String>(data)): 
print(c); // Reading is OK 
//! c.add("one"); // Can't change it 


List<String> a = Collections. unmodifiableList( 
new ArrayList<String>(data)) ; 

ListIterator<String> lit = a.listIterator(); 

print(lit.next()): // Reading is OK 

//! Vit.add("one"); // Can't change it 


Set<String> s = Collections.unmodifiableSet ( 
new HashSet<String>(data)); 

print(s); // Reading is OK 

//! s.add("one"); // Can't change it 


// For a Sortedset: 
Set<String> ss = Collections. unmodifiableSortedset( 
new TreeSet<String>(data)) ; 


Map<String, String> m = Collections.unmodifiableMap( 
new HashMap<String, String> (Countries. capitals(6)))s 

print(m); // Reading is OK 

7/1 m.put("Ralph", "Howdy! ") ; 


// For a SortedMap: 
Map<String, String> sm = 
Collections. unmodi fiableSortedMap( 
new TreeMap<String, String> (Countries. capitals(6))); 





} 
} /* Output: 
[ALGERIA, ANGOLA, BENIN, BOTSWANA, BULGARIA, BURKINA FASO} 
ALGERIA 

[BULGARIA, BURKINA FASO, BOTSWANA, BENIN, ANGOLA, ALGERIA) 
{BULGARIA=Sofia, BURKINA FASO=Quagadougou, 
BOTSWANA=Gaberone, BENIN=Porto-Novo, ANGOLA=Luanda, 





对 特定 类 型 的 “不 可 修改 的 ”方法 的 调用 并 不 会 产生 编译 时 的 检查 ， 但 是 转换 完成 后 ， 任 
何 会 改变 容器 内 容 的 操作 都 会 引起 UnsupportedOperationException 异 常 。 

无 论 哪 一 种 情况 ， 在 将 容器 设 为 只 读 之 前 ， 必 须 填 入 有 意义 的 数据 。 装 载 数据 后 ， 就 应 该 
使 用 “不 可 修改 的 ”方法 返回 的 引用 去 替换 掉 原本 的 引用 。 这 样 ， 就 不 用 担心 无 意 中 修改 了 只 
读 的 内 容 。 另 一 方面 ， 此 方法 允许 你 保留 一 份 可 修改 的 容器 ， 作 为 类 的 private 成 员 ， 然 后 通过 
某 个 方法 调用 返回 对 该 容器 的 “只 读 ” 的 引用 。 这 样 以 来 ， 就 只 有 你 可 以 修改 容器 的 内 容 ， 而 
别人 只 能 读 取 。 

17.11.3 Collection 或 Map 的 同步 控制 

关键 字 synchronized 是 多 线程 议题 中 的 重要 部 分 , 第 21 章 将 讨论 这 种 较为 复杂 的 主题 。 这 里 ， 
我 只 提醒 读者 注意 ，Collections 类 有 办 法 能 够 自动 同步 整个 容器 。 其 语法 与 “不 可 修改 的 ”方法 
相似 : 


/1: containers/Synchronization. java 
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// Using the Collections.synchronized methods. 
import java.util.*; 


public class Synchronization { 
public static void main(String{] args) { 
Collection<String> c = 
Collections. synchronizedCollection( 
new ArrayList<String>()); 
List<String> list = Collections. synchronizedList( 
new ArrayList<String>()): 
SetsString> s = Collections. synchronizedSet(_ 
new HashSet<String>()); 
Set<String> ss = Collections. synchronizedSortedSet( 
new TreeSet<String>()); 
Map<String, String> m = Collections. synchronizedMap( 
new HashMap<String, String>()); 
Map<String.String> sm = 
Collections. synchronizedSortedMap( 
new TreeMap<String,String>()); 


) 

} :~ 

最 好 是 如 上 所 示 ， 直 接 将 新 生成 的 容器 传递 给 了 适当 的 “同步 ”方法 ， 这 样 做 就 不 会 有 任 
何 机 会 暴露 出 不 同步 的 版 本 。 ‘ 

快速 报错 

Java 容器 有 一 种 保护 机 制 ， 能 够 防止 多 个 进程 同时 修改 同一 个 容器 的 内 容 。 如 果 在 你 选 代 
遍历 某 个 容器 的 过 程 中 ， 另 一 个 进程 介 信 其中， 并且 插入 、 删 除 或 修改 此 容器 内 的 某 个 对 象 ， 
那么 就 会 出 现 问题 : 也 许 迭 代 过 程 已 经 处 理 过 容器 中 的 该 元 素 了 ， 也 许 还 没 处 理 ， 也 许 在 调用 
size() 之 后 容器 的 尺寸 收缩 了 一 一 还 有 许多 灾难 情景 。Java 容 器 类 类 库 采 用 快速 报错 (fail-fast) 
机 制 。 它 会 探查 容器 上 的 任何 除了 你 的 进程 所 进行 的 操作 以 外 的 所 有 变化 ， 一 旦 它 发 现 其 他 进 
程 修改 了 容器 ， 就 会 立刻 抛 出 ConcurrentModificationException 异 常 。 这 就 是 “快速 报错 ”的 
意思 一 即 ， 不 是 使 用 复杂 的 算法 在 事后 来 检查 问题 。 

很 容易 就 可 以 看 出 “快速 报错 ”机 制 的 工作 原理 : 只 需 创建 一 个 迭代 器 ， 然 后 向 选 代 器 所 
指向 的 Colleetion 添 加 点 什么 ， 就 像 这 样 : 


//: containers/FailFast.java 
// Demonstrates the "fail-fast" behavior. 
import java.util.*; 





public class FailFast { 
public static void main(String[] args) { 
Collection<String> c = new ArrayList<String>(): 
Iterator<String> it = c.iterator(); 
c.add("An object"); N, 
try { 
String s = it.next(): 
} catch(ConcurrentModificationException e) { 
System. out.printin(e) ; A 
} 
} 
} /* Output: 
java.util.ConcurrentModificationException 
"HT: 


程序 运行 时 发 生 了 异常 ， 因 为 在 容器 取得 迭代 器 之 后 ， 又 有 东西 被 放 入 到 了 该 容器 中 。 当 
程序 的 不 同 部 分 修改 同一 个 容器 时 ， 就 可 能 导致 容器 的 状态 不 一 致 ， 所 以 ;此 异常 提醒 你 ， 应 
该 修改 代码 。 在 此 例 中 ， 应 该 在 添加 完 所 有 的 元 素 之 后 ， 再 获取 迁 代 器 。 

ConcurrentHashMap、CopyOnWriteArrayList 和 CopyOnWriteArraySet 都 使 用 了 可 以 避免 














890| 
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ConcurrentModificationException 的 技术 。 
17.12 持 有 引用 


java.lang.ref 类 库 包含 了 一 组 类 ， 这 些 类 为 垃圾 回收 提供 了 更 大 的 灵活 性 。 当 存在 可 能 会 耗 
尽 内 存 的 大 对 象 的 时 候 ， 这 些 类 显得 特别 有 用 。 有 三 个 继承 自 抽象 类 Reference 的 类 : 
SoftReference、WeakReference 和 PhantomReference。 当 垃圾 回收 器 正在 考察 的 对 象 只 能 通过 
某 个 Reference 对 象 才 “可 获得 ”时 ， 上 述 这 些 不 同 的 派生 类 为 垃圾 回收 器 提供 了 不 同 级 别 的 间 
接 性 指示 。 

对 象 是 可 获得 的 《reachable) ， 是 指 此 对 象 可 在 程序 中 的 某 处 找到 。 这 意味 着 你 在 栈 中 有 一 
个 普通 的 引用 ， 而 它 正 指向 此 对 象 ， 也 可 能 是 你 的 引用 指向 某 个 对 象 ， 而 那个 对 象 含有 另 一 个 
引用 指向 正在 讨论 的 对 象 ， 也 可 能 有 更 多 的 中 间 链 接 。 如 果 一 个 对 象 是 “可 获得 的 "， 垃 圾 回收 
器 就 不 能 释放 它 ， 因 为 它 仍然 为 你 的 程序 所 用 。 如 果 一 个 对 象 不 是 “可 获得 的 "， 那 么 你 的 程序 
将 无 法 使 用 到 它 ， 所 以 将 其 回收 是 安全 的 。 

如 果 想 继续 持 有 对 某 个 对 象 的 引用 ， 和 希望 以 后 还 能 够 访问 到 该 对 象 ， 但 是 也 希望 能 够 允许 
垃圾 回收 器 释放 它 ， 这 时 就 应 该 使 用 Reference 对 象 。 这 样 ， 你 可 以 继续 使 用 该 对 象 ， 而 在 内 存 
消耗 至 尽 的 时 候 又 允许 释放 该 对 象 。 

以 Reference 对 象 作为 你 和 普通 引用 之 间 的 媒介 (代理 )， 另 外 ， 一 定 不 能 有 普通 的 引用 指向 
那个 对 象 ， 这 样 就 能 达到 上 述 目 的 。( 普 通 的 引用 指 没 有 经 Reference 对 象 包装 过 的 引用 。) 如 果 
垃圾 回收 器 发 现 某 个 对 象 通过 普通 引用 是 可 获得 的 ， 该 对 象 就 不 会 被 释放 。 

SoftReference、WeakReference 和 PhantomReference 由 强 到 弱 排 列 ， 对 应 不 同 级 别 的 “可 
获得 性 ”。Softreference 用 以 实现 内 存 敏感 的 高 速 缓存 。Weak reference 是 为 实现 “规范 映射 ” 
(canonicalizing mappings) 而 设计 的 ， 它 不 妨碍 垃圾 回收 器 回收 映射 的 “ 键 ”( 或 “ 值 ")。“ 规 
范 映射 ”中 对 象 的 实例 可 以 在 程序 的 多 处 被 同时 使 用 ， 以 节省 存储 空间 。Phantomreference 用 
以 调度 回收 前 的 清理 工作 ， 它 比 Java 终 止 机 制 更 灵活 。 

使 用 SoftReference 和 WeakReference 时 ， 可 以 选择 是 否 要 将 它们 放 入 ReferenceQueue (用 作 
“回收 前 清理 工作 ”的 工具 )。 而 PhantomReference 只 能 依赖 于 ReferenceQueue。 下 面 是 一 个 简 
单 的 示例 : 


//: contafners/References.java 
// Demonstrates Reference objects 
import java. lang. ref.*; 

import java.util.*; 


class VeryBig { 
private static final int SIZE = 10000; 
private long(] la = new long[SIZE] ; 
private String ident; 
public VeryBig(String id) { ident = id: } 
public String toString() { return ident; } 
protected void finalize() { 

System.out.printin("Finalizing ”+ ident); 

} 

} 


public class References { 
private static ReferenceQueue<VeryBig> rq = 
new ReferenceQueue<VeryBig>() ; 
public static void checkQueue() { 
Reference<? extends VeryBig> ing = rq.poll(); 
if (ing != null) 
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System.out.printin("In queue: ”+ ing.get()): 


public static void main(String[] args) { 
int size = 10; 
// Or, choose size via the command line: 
if(args.length > 9) 
size = new Integer (args[9}); 
LinkedList<SoftReference<VeryBig>> sa = ` 
new LinkedList<SoftReference<VeryBig>>(); 
for(int 1 =; i < size; i++) { 
sa.add(new SoftReference<VeryBig>( 
new VeryBig("Soft " + i), rq)); 
System.out.printin("Just created: ”+ sa.getLast()); 
checkQueue() ; 
} 
LinkedList<WeakReference<VeryBig>> wa = 
new LinkedList<WeakReference<VeryBig>>(); 
for(int i = 0; 1 < size; i++) { 
wa.add(new WeakReference<VeryBig>( 
new VeryBig("Weak " + i), rq)); 
System.out.printin("Just created: ”+ wa.getLast()): 
checkQueue() ; 


SoftReference<VeryBig> s = 
new Sof tReference<VeryBig> (new VeryBig("Soft")); 
WeakReference<VeryBig> w = 
new WeakReference<VeryBig>(new VeryBig("Weak")); 
System.gc(); 
LinkedList<PhantomReference<VeryBig>> pa = 
new LinkedList<PhantomReference<VeryBig>>(); 
for(int i = @; 1 < size; i++) { 
pa.add(new PhantomReference<VeryBig>( 
new VeryBig("Phantom "+ i), rq)); 
System.out.printIn("Just created: ”+ pa.getLast()); 
checkQueue() ; 
} 


i 

运行 此 程序 可 以 看 到 (将 输出 重 定向 到 一 个 文本 文件 中 ， 便 可 以 查看 分 页 的 输出 )， 尽 管 还 
要 通过 Reference 对 象 访问 那些 对 象 (使 用 get0 取 得 实际 的 对 象 引用 ) ， 但 对 象 还 是 被 垃圾 回收 器 
回收 了 。 还 可 以 看 到 ，ReferenceQueue 总 是 生成 一 个 包含 nul 对 象 的 Reference。 要 利用 此 机 制 ， 
可 以 继承 特定 的 Reference 类 ， 然 后 为 这 个 新 类 添加 一 些 更 有 用 的 方法 。 
17.12.1 WeakHashMap 

容器 类 中 有 一 种 特殊 的 Map， 即 WeakHashMap， 它 被 用 来 保存 WeakReference。 它 使 得 规 
范 映 射 更 易于 使 用 。 在 这 种 映射 中 ， 每 个 值 只 保存 一 份 实例 以 节省 存储 空间 。 当 程序 需要 那个 
“ 值 ”的 时 候 ， 便 在 映射 中 查询 现 有 的 对 象 ， 然 后 使 用 它 (而 不 是 重新 再 创建 )。 映 射 可 将 值 作 
为 其 初始 化 中 的 一 部 分 ， 不 过 通常 是 在 需要 的 时 候 才 生成 “ 值 "。 

这 是 一 种 节约 存储 空间 的 技术 ， 因 为 WeakHashMap 人 允许 垃圾 回收 器 自动 清理 键 和 值 ， 所 以 
它 显得 十 分 便利 。 对 于 向 WeakHashMap 添 加 键 和 值 的 操作 ， 则 没有 什么 特殊 要 求 。 映 射 会 自动 
使 用 WeakReference 包 装 它们 。 允 许 清理 元 素 的 触发 条 件 是 ， 不 再 需要 此 键 了 ， 如 下 所 示 : 

11: containers/CanonicatMapping. java 


// Demonstrates WeakHashMap. 
import java.util.*; 


class Element { 
private String ident; 
public Element (String id) { ident = id: } 
public String toString() { return ident; } 
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public int hashCode() { return ident.hashCode(); } 
public boolean equals(Object r) { 
return r instanceof Element && 
ident equals (((Element)r). ident); 


} 
protected void finalize() { 
System.out.printin("Finalizing " + 
getClass().getSimpleName() +" ”+ ident); 
} 
} 


class Key extends Element { 
public Key(String id) { super(id); } 


class Value extends Element { 
public Value(String id) { super(id); } 
} 


public class CanonicalMapping { 
public static void main(String[] args) { 

int size = 1000; 
// Or, choose size via the command line: 
if(args.length > 0) 

size = new Integer (args{0]); 
Key[] keys = new Key[size]; 
WeakHashMap<Key,Value> map = 

new WeakHashMap<Key ,Value>(): 
for(int i = @; 1 < size; i++) { 








Key k new Key(Integer.toString(i)); 
Value v = Value(Integer.toString(i)): 
if(i % 3 ) 

keys[i] = k; // Save as “real” references 








map.put(k, v); 
statio; 
} hs (Execute to see output) *///:~ 
如 同 本 章 前 面 所 述 ，Key 类 必须 有 hashCode0 和 equals0， 因 为 在 散 列 数据 结构 中 ， 它 被 用 
作 键 。 有 关 hashCode0 的 主题 在 本 章 前 面部 分 已 经 描述 过 了 。 
” ”运行 此 程序 ， 会 看 到 垃圾 回收 器 每 隔 三 个 键 就 跳 过 一 个 ， 因 为 指向 那个 键 的 普通 引用 被 存 
人 了 keys 数 组 ， 所 以 那些 对 象 不 能 被 垃圾 回收 器 回收 。 


17.13 Java 1.0/1.1 的 容器 


很 不 幸 ， 许 多 老 的 代码 是 使 用 Java 1.0/1.1 的 容器 写成 的 ， 甚 至 有 些 新 的 程序 也 使 用 了 这 些 
类 。 因 此 ， 虽 然 在 写 新 的 程序 时 ， 决 不 应 该 使 用 旧 的 容器 ， 但 你 仍然 应 该 了 解 它们 。 不 过 旧 容 
器 功能 有 限 ， 所 以 对 它们 也 没 太 多 可 说 的 。 毕 竞 它 们 都 过 时 了 ， 所 以 我 也 不 想 强调 某 些 设计 有 
多 精 糕 。 

17.13.1 Vector 和 Enumeration 

在 Java 1.0/1.1 中 ，Vector 是 唯一 可 以 自我 扩展 的 序列 ， 所 以 它 被 大 量 使 用 。 它 的 缺点 多 到 这 
里 都 难以 描述 (可 以 参见 本 书 的 第 1 版 ， 可 从 www.MindView.net 免 费 下 载 )。 基 本 上 ， 可 将 其 看 
作 ArrayList， 但 是 具有 又 长 又 难 记 的 方法 名 。 在 订正 过 的 Java 容 器 类 类 库 中 ，Vector 被 改造 过 ， 
可 将 其 归 类 为 Collection 和 List。 这 样 做 有 点 不 妥当 ， 可 能 会 让 人 误会 Vector 变 得 好 用 了 ， 实 际 上 
这 样 做 只 是 为 了 支持 Java 2 之 前 的 代码 。 

Java 1.0/1.1 版 的 迭代 器 发 明了 一 个 新 名 字 一 一 枚 举 ， 取 代 了 为 人 熟知 的 术语 GRE). ie 
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Enumeration 接口 比 Iterator 小 ， 只 有 两 个 名 字 很 长 的 方法 : 一 个 为 boolean hasMoreElements(), 
如 果 此 枚 举 包含 更 多 的 元 素 ， 该 方法 就 返回 true， 另 一 个 为 Object nextElement0， 该 方法 返回 
此 枚 举 中 的 下 一 个 元 素 (如 果 还 有 的 话 )， 否 则 抛 出 异常 。 

Enumeration 只 是 接口 而 不 是 实现 ， 所 以 有 时 新 的 类 库 仍然 使 用 了 旧 的 Enumeration， 这 令 
人 十 分 遗憾 ,但 通常 不 会 造成 伤害 。 虽然 在 你 的 代码 中 应 该 尽量 使 用 Iterator， 但 也 得 有 所 准备 ， 
类 库 可 能 会 返回 给 你 一 个 Enumeration。 

此 外 ， 还 可 以 通过 使 用 Collections.enumeration0 方 法 来 从 Collection 生 成 一 个 Enumeration， 
见 下 面 的 例子 : 


//: containers/Enumerations. java 
// Java 1.0/1.1 Vector and Enumeration. 
import java.util.*; 

import net.mindview.util.*; 


public class Enumerations { 
public static void main(String{] args) { 
Vector<String> v = 
new Vector<String> (Countries .names (10)) ; 
Enumeration<String> e = v.elements(); 
whi le(e. hasMoreElements()) 
System.out.print(e.nextElement() + ", "); 


// Produce an Enumeration from a Collection: 
e = Collections.enumeration(new ArrayList<String>()); 


} k Output: 

ALGERIA, ANGOLA, BENIN, BOTSWANA, BULGARIA, BURKINA FASO, 

Sonor: CAMEROON, CAPE VERDE, CENTRAL AFRICAN REPUBLIC, > 

可 以 调用 elements0 生 成 Enumeration， 然 后 使 用 它 进行 前 序 遍 历 。 最 后 一 行 代码 创建 了 一 
个 ArrayList， 并 且 使 用 enumeration0 将 ArrayList 的 Iterator 转 换 成 了 Enumeration。 这 样 ， 即 使 
有 需要 Enumeration 的 旧 代 码 ， 你 仍然 可 以 使 用 新 容器 。 
17.13.2 Hashtable 

正如 在 前 面 性 能 比较 中 所 看 到 的 ,基本 的 Hashtable 与 HashMap 很 相似 , 甚至 方法 名 也 相似 。 
所 以 ， 在 新 的 程序 中 ， 没 有 理由 再 使 用 Hashtable 而 不 用 HashMap。 
17.13.3 Stack 

前 面 在 使 用 LinkedList 时 ， 已 经 介绍 过 “ 栈 ” 的 概念 。Java 1.0/1.1 的 Stack 很 奇怪 ， 竟 然 不 
是 用 Vector 来 构建 Stack， 而 是 继承 Vector。 所 以 它 拥有 Vector 所 有 的 特点 和 行为 ， 再 加 上 一 些 
额外 的 Stack 行 为 。 很 难 了 解 设计 者 是 否 意识 到 这 样 做 特别 有 用 处 ， 或 者 只 是 一 个 幼稚 的 设计 。 
唯一 清楚 的 是 ， 在 匆忙 发 布 之 前 它 没有 经 过 仔细 审查 ， 因 此 这 个 精 糕 的 设计 仍然 挂 在 这 里 (但 
是 你 永远 都 不 应 该 使 用 它 ) 。 

这 里 是 Stack 的 一 个 简单 示例 ， 将 enum 中 的 每 个 String 表 示 压 人 Stack。 它 还 展示 了 你 可 以 
如 何方 便 地 将 LinkedList， 或 者 在 第 11 章 中 创建 的 Stack 类 用 作 栈 : 


//: containers/Stacks.java 

// Demonstration of Stack Class. 

import java.util.*; 

import static net.mindview.util.Print.*; 

enum Month { JANUARY, FEBRUARY, MARCH, APRIL, MAY, JUNE. 
JULY, AUGUST, SEPTEMBER, OCTOBER, NOVEMBER } 


public class Stacks { 
public static Void main(String[] args) { 
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Stack<String> stack = new StacksString>(); 
for (Month m : Month. values()) 
stack. push(m. toString()); 
print("stack = ”+ stack): 
// Treating a stack as a Vector: 
stack. addElement ("The last Line"); 
print("element 5 = " + stack.elementAt(5)); 
«print ("popping elements:"): 
while(!stack.empty()) 
printnb(stack.pop() + 





// Using a LinkedList as a Stack: 
LinkedList<String> lstack = new LinkedList<String>(); 
for(Month m : Month. values()) 
Istack. addFirst(m.toString()); 
print("Istack = " + Lstack); 
while(!lstack. isEmpty ()) 
printnb(Lstack.removeFirst() + " "); 


// Using the Stack class from 
// the Holding Your Objects Chapter: 
net .mindview.util.Stack<String> stack2 = 
new net.mindview.util.Stack<String>(); 
for(Month m : Month.values()) 
stack2. push(m.toString()) ; 
print("stack2 = ”+ stack2); 
while(!stack2.empty()) 
printnb(stack2.pop() +" "); 


} 
} /* Output: 
stack = [JANUARY, FEBRUARY, MARCH, APRIL’, MAY, JUNE, JULY; Y 
AUGUST, SEPTEMBER, OCTOBER, NOVEMBER) 
element 5 = JUNE 
popping elements: 
The last line NOVEMBER OCTOBER SEPTEMBER AUGUST JULY JUNE 
MAY APRIL MARCH FEBRUARY JANUARY Istack = (NOVEMBER, 
OCTOBER, SEPTEMBER, AUGUST, JULY, JUNE, MAY, APRIL, MARCH, 
FEBRUARY, JANUARY] 
NOVEMBER OCTOBER SEPTEMBER AUGUST JULY JUNE MAY APRIL MARCH 
FEBRUARY JANUARY stack2 = (NOVEMBER, OCTOBER, SEPTEMBER, 
AUGUST, JULY, JUNE, MAY, APRIL, MARCH, FEBRUARY, JANUARY) 
NOVEMBER OCTOBER SEPTEMBER AUGUST JULY JUNE MAY APRIL MARCH 
FEBRUARY JANUARY 
Wh 


String 表 示 是 从 Month enum 常 量 中 生成 的 ， 用 push0 插 入 Stack， 然 后 再 从 栈 的 顶端 弹出 来 
(用 popO)。 这 里 要 特别 强调 : 可 以 在 Stack 对 象 上 执行 Vector 的 操作 。 这 不 会 有 任何 问题 ， 因 为 
继承 的 作用 使 得 Stack 是 一 个 Veetor， 因 此 所 有 可 以 对 Vector 执行 的 操作 ， 都 可 以 对 Stack 执 行 ， 
例如 elementAt0。 

前 面 曾经 说 过 ， 如 果 需 要 栈 的 行为 ， 应 该 使 用 LinkedList, 或 者 从 LinkeqList 类 中 创建 的 
net.mindview.util.Stack 类 。 

17.13.4 BitSet 

如 果 想 要 高 效率 地 存储 大 量 “ 开 / 关 ” 信 息 ，BitSet 是 很 好 的 选择 。 不 过 它 的 效率 仅 是 对 空 
间 而 言 ， 如 果 需 要 高 效 的 访问 时 间 ，BitSet 比 本 地 数组 稍 慢 一 上 

此 外 ，BitSet 的 最 小 容量 是 long: 64 位 。 如 果 存 储 的 内 容 比较 小 ~ 例如 8 位 ， 那 么 BitSet 就 浪 
费 了 一 些 空间 。 因 此 如 果 空 间 对 你 很 重要 ， 最 好 撰写 自己 的 类 ， 或 者 直接 采用 数组 来 存储 你 的 
标志 信息 (只 有 在 创建 包含 开关 信息 列表 的 大 量 对 象 ， 并 且 促 使 你 做 出 决定 的 依据 仅仅 是 性 能 
和 其 他 度量 因素 时 ， 才 属于 这 种 情况 。 如 果 你 做 出 这 个 决定 只 是 因为 你 认为 某 些 对 象 太 大 了 ， 
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那么 你 最 终 会 产生 不 需要 的 复杂 性 ， 并 会 浪费 掉 大 量 的 时 间 )。 

普通 的 容器 都 会 随 着 元 素 的 加 入 而 扩充 其 容量 ，BitSet 也 是 。 以 下 示范 了 BitSet 是 如 何 工 
YERI: 

//: containers/Bits.java 

// Demonstration of BitSet. 


import java.util.*; 
import static net 





indview.util.Print.*; 





public class Bits { 
public static void printBitSet(Bitset b) { 
print("bits: " + b); 
StringBuilder bbits = new StringBuilder (); 
for(int j = 0; j < b.size() ; j++) 
bbits.append(b.get(j) ? "1" : "@"); 
print("bit pattern: " + bbits); 
} 
public static void main(String(] args) { 
Random rand = new Random(47) ; 
// Take the LSB of nextInt(): 
byte bt = (byte)rand.nextInt(); 
BitSet bb = new Bitset( 
for(int i= 7; i >= 8 
if(((1 << 4) & bt) != 6) 
bb.set(i); 
else 
bb.clear(1); 
print("byte value: 
printBitSet (bb) ; 











short st = (short)rand.nextInt(); 
BitSet bs = new BitSet(); 
for(int 1 = 15; 1 >= 6; i--) 
1f(((1 << i) & st) != @) 
bs.set(i): 
else 
bs.clear(i); 
print("short value: " + st); 
printBitset (bs) ; 


int it = rand.nextInt(); 
BitSet bi = new BitSet( 
for(int 1 = 31; 1 >= 0; 
if(((L << i) & it) != 8) 
bi.set(i); 
else 
bi.clear(i); . 
print("int value: " + it); 
printBitSet (bi); 








// Test bitsets >= 64 bits: 
BitSet b127 = new Bitset( 
b127. set (127); 
print("set bit 127: "+ b127); 
BitSet b255 = new BitSet(65); 
b255. set (255); 
print("set bit 255: * + b255); 
BitSet b1023 = new BitSet(512); 
b1023. set (1023) ; 
b1023. set (1024) ; 
print("set bit 1023: ”+ b1623); 
} 

} /* Output: 

byte value: -107 

bits: {0, 2, 4, 7} 

bit pattern: 











897, 














898 











524 ” IÈ 











[899] 














(900| 








101010010000000000000000000000000000000000000000000000900009 
00000 


short value: 1302 2 = 


bits: {1, 2, 4, 8, 10} 

bit pattern: 
61161666161666666668668666686866666666666666868866666668666 
90000 

int value: -2014573909 

bits: (8, 1, 3, 5, 7, 9, 11, 18, 19, 21, 22, 23, 24, 25,. 

26, 31) 

bit pattern: 
11616161616166666611611111166961966696666866966666666666699 
00000 

set bit 127: {127} 

set bit 255: {255} 

set bit 1023: {1023, 1024} x ý 
Whim i 


随机 数 发 生 器 被 用 来 生成 随机 的 byte、short 和 int， 每 一 个 都 被 转换 为 BitSet 中 相应 的 位 模 
式 。 因 为 BitSet 是 64 位 的 ， 所 以 任何 生成 的 随机 数 都 不 会 导致 BitSet 扩 充 容量 。 然 后 创建 了 一 个 
更 大 的 BitSet。 你 可 以 看 到 ，BitSet 在 必要 时 会 进行 扩充 。 

如 果 拥 有 一 个 可 以 命名 的 固定 的 标志 集合 ， 那 么 EnumSet (查看 第 19 章 ) 与 BitSet 相 比 ， 
通常 是 一 种 更 好 的 选择 ， 因 为 EnumSet 人 允许 你 按照 名 字 而 不 是 数字 位 的 位 置 进行 操作 ， 因 此 可 
以 减少 错误 。EnuimSet 还 可 以 防止 你 因 不 注意 而 添加 新 的 标志 位 置 ， 这 种 行为 能 够 引发 严重 的 、 
难以 发 现 的 缺陷 。 你 应 该 使 用 BitSet 而 不 是 EnumSet 的 理由 只 包括 : 只 有 在 运行 时 才 知 道 需要 
多 少 个 标志 ， 对 标志 命名 不 合理 ， 需 要 BitSet 中 的 某 种 特殊 操作 (查看 BitSet 和 EnumSet 的 JDK 
文档 )。 + 


17.14 总 结 


可 以 证 明 ， 容 器 类 库 对 于 面向 对 象 语言 来 说 是 最 重要 的 类 库 。 大 多 数 编程 工作 对 容器 的 使 
用 比 对 其 他 类 库 中 的 构件 都 要 多 。 某 些 语言 (例如 Python) 甚至 包含 内 建 的 基本 容器 构件 (A 
表 、 映 射 表 和 集 ) 。 

正如 你 在 第 11 章 中 所 看 到 的 ， 通 过 使 用 容器 ， 无 须 费 力 ， 就 可 以 完成 大 量 非常 有 趣 的 操作 。 
但 是 ， 在 某 些 时 候 ， 你 必须 更 多 地 了 解 容 器 ， 以 便 正确 地 使 用 它们 。 特 别 是 ， 你 必须 对 散 列 操 
作 有 足够 的 了 解 ， 从 而 能 够 编写 自己 的 hashCode0 方 法 (并 且 你 必须 知道 何 时 需要 这 么 做 )， 你 
还 必须 对 各 种 不 同 的 容器 实现 有 足够 的 了 解 ， 这 样 才 能 够 为 你 的 需要 进行 恰当 的 选择 。 本 章 覆 
盖 了 有 关 容 器 类 库 的 这 些 概念 ， 并 讨论 了 其 他 有 用 的 细节 。 至 此 ， 你 应 该 已 经 为 在 每 天 的 编程 
任务 中 使 用 Java 容 器 做 好 了 充足 的 准备 。 

容器 类 库 的 设计 非常 艰难 (大 多 数 类 库 设计 问题 都 是 如 此 )。 在 C++ 中 ， 用 许多 不 同 的 类 覆 
盖 了 容器 类 的 基础 。 这 与 C++ 容 器 类 之 前 的 可 用 情况 (无 任何 类 可 用 ) 相 比 是 一 种 进步 ， 但 是 
它 没有 被 很 好 地 转译 到 Java 中 。 在 另 一 个 极端 情况 中 ， 我 看 到 过 容器 类 库 由 单一 的 类 构成 ， 即 
Container， 它 同时 起 到 了 线性 序列 和 关联 数组 的 作用 。Java 容 器 类 库 在 这 二 者 之 间 达 到 了 一 种 平 
i: 具有 成 熟 的 容器 类 库 应 该 具有 的 完备 的 功能 ， 但 是 比 C++ 容器 类 和 其 他 类 似 的 容器 类 库 易 
于 学 习 和 使 用 。 这 样 产生 的 结果 在 若干 方面 看 起 来 都 有 些 奇异 ， 与 早期 java 类 库 中 所 作 的 某 些 
决策 不 同 ， 这 些 奇异 性 不 是 偶然 的 ， 而 是 基于 复杂 性 的 利 浆 而 仔细 权衡 的 产物 。 

所 选 习题 的 答案 都 可 以 在 名 为 The Thinking in Java Annotated Solution Guide 的 电子 文档 中 找 
到 ， 读 者 可 以 从 wwwMindView.net 购 买 此 文档 。 


’ 





188 Java I/O 系 统 


对 程序 语言 的 设计 者 来 说 ， 创 建 一 个 好 的 输入 /输出 (I/O) 系统 是 一 项 艰难 的 任务 。 

现 有 的 大 量 不 同方 案 已 经 说 明了 这 一 点 。 挑 战 似乎 来 自 于 要 涵盖 所 有 的 可 能 性 。 不 仅 存在 
各 种 IO 源 端 和 想 要 与 之 通信 的 接收 端 (文件 、 控 制 台 、 网 络 链接 等 )， 而 且 还 需要 以 多 种 不 同 
的 方式 与 它们 进行 通信 (顺序 、 随 机 存 取 、 缓 冲 、 二 进 制 、 按 字符 、 按 行 、 按 字 等 )。 

Java 类 库 的 设计 者 通过 创建 大 量 的 类 来 解决 这 个 难题 。 一 开始 ， 可 能 会 对 Java VO 系统 提供 
了 如 此 多 的 类 而 感到 不 知 所 措 (具有 讽刺 意味 的 是 ，Java WO 设计 的 初衷 是 为 了 避免 过 多 的 类 )。 
自从 Java 1.0 版 本 以 来 ，Java 的 MO 类 库 发 生 了 明显 改变 ， 在 原来 面向 字 节 的 类 中 添加 了 面向 字符 
和 基于 Unicode 的 类 。 在 JDK 1.4 中 ， 添 加 了 nio 类 (对 于 “新 MO” 来 说 ， 这 是 一 个 从 现在 起 我 们 
将 要 使 用 若干 年 的 名 称 ， 即 使 它们 在 JDK1.4 中 就 已 经 被 引入 了 ， 因 此 它们 已 经 “ 旧 ” 了 ) 添加 
进来 是 为 了 改进 性 能 及 功能 。 因 此 ， 在 充分 理解 Java W/O 系统 以 便 正确 地 运用 之 前 ， 我 们 需要 学 
习 相 当 数 量 的 类 。 另 外 ， 很 有 必要 理解 WO 类 库 的 演化 过 程 ， 即 使 我 们 的 第 一 反应 是 “不 要 用 历 
史 打 扰 我 ， 只 需 告 诉 我 怎么 用 。” 问 题 是 ， 如 果 缺 乏 历史 的 眼光 ， 很 快 我 们 就 会 对 什么 时 候 该 使 
用 哪些 类 ， 以 及 什么 时 候 不 该 使 用 它们 而 感到 迷惑 。 

本 章 就 介绍 Java 标 准 类 库 中 各 种 各 样 的 类 以 及 它们 的 用 法 。 


18.1 File 类 


在 学 习 那 些 真正 用 于 在 流 中 读 写 数据 的 类 之 前 ， 让 我 们 先 看 一 个 实用 类 库 工 具 ， 它 可 以 帮 
助 我 们 处 理 文件 目录 问题 。 

File (文件 ) 类 这 个 名 字 有 一 定 的 误导 性 ， 我 们 可 能 会 认为 它 指 代 的 是 文件 ， 实 际 上 却 并 非 
如 此 。 它 既 能 代表 一 个 特定 文件 的 名 称 ， 又 能 代表 一 个 目录 下 的 一 组 文件 的 名 称 。 如 果 它 指 的 
是 一 个 文件 集 ， 我 们 就 可 以 对 此 集合 调用 list0 方 法 ， 这 个 方法 会 返回 一 个 字符 数组 。 我 们 很 容 
易 就 可 以 理解 返回 的 是 一 个 数组 而 不 是 某 个 更 具 灵 活性 的 类 容器 ， 因 为 元 素 的 个 数 是 固定 的 ， 
所 以 如 果 我 们 想 取得 不 同 的 目录 列表 ， 只 需要 再 创建 一 个 不 同 的 File 对 象 就 可 以 了 。 实 际 上 ， 
FilePath (文件 路 径 ) 对 这 个 类 来 说 是 个 更 好 的 名 字 。 本 节 举 例 示范 了 这 个 类 的 用 法 ， 包 括 了 与 
它 相 关 的 FilenameFilter 接 口 。 

18.1.1 目录 列表 器 

假设 我 们 想 查看 一 个 目录 列表 ， 可 以 用 两 种 方法 来 使 用 File 对 象 。 如 果 我 们 调用 不 带 参数 的 
list0 方 法 ， 便 可 以 获得 此 File 对 象 包含 的 全 部 列表 。 然 而 ， 如 果 我 们 想 获得 一 个 受 限 列表 ， 例 如 ， 
想得到 所 有 扩展 名 为 .java 的 文件 ， 那 么 我 们 就 要 用 到 “目录 过 滤器 ”"， 这 个 类 会 告诉 我 们 怎样 显 
示 符 合 条 件 的 File 对 象 。 

下 面 是 一 个 示例 ， 注 意 ， 通 过 使 用 java.utils.Arrays.sortO 和 String.CASE_INSENSITIVE. 
ORDERComparator， 可 以 很 容易 地 对 结果 进行 排序 ( 按 字 母 顺 序 )。 


1/: 10/DirList.java 

// Display a directory Listing using regular expressions. 
11 {Args: "D.*\.java"} 

import java.util.regex.*; 

import java.io.*; 
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import java.util.*; 


public class DirList { 
public static void main(Stringl] args) { 
File path = new File(*."); 
String(] list; 
if(args.length == 0) 
List = path. listQ; 
else 
Vist = path. list(new DirFilter (args{0])); 
Arrays.sort(List, String-CASE_INSENSITIVE_ORDER) ; 
for(String dirltem : list) 
System. out.printIn(dirltem) ; 
} 
$ 


class DirFilter implements FilenameFilter { 
private Pattern pattern; 
public DirFilter (String regex) { 
pattern = Pattern.compile(regex) ; 


$ 
public boolean accept(File dir, String name) { 
return pattern.matcher (name) .matches(): 


} 
} /* Output: 
DirectoryDemo. java 
DirList.java 
DirList2. java 
DirList3. java 
“N~ 


这 里 ，DirFilter 类 “实现 ”了 FilenameFilter 接 口 。 请 注意 FilenameFilter 接 口 是 多 么 的 简单 


public interface FilenameFilter ( 
boolean accept(File dir, String name); 
} 


DirFilter 这 个 类 存在 的 唯一 原因 就 是 将 accept0 方 法 。 创 建 这 个 类 的 目的 在 于 把 accept0 方 法 
提供 给 list0 使 用 ， 使 list0 可 以 回调 accept0， 进 而 以 决定 哪些 文件 包含 在 列表 中 。 因 此 ， 这 种 结 
构 也 常常 称 为 回调 。 更 具体 地 说 ， 这 是 一 个 策略 模式 的 例子 ， 因 为 list0 实 现 了 基本 的 功能 ， 而 
且 按 照 FilenameFilter 的 形式 提供 了 这 个 策略 ， 以 便 完善 list0 在 提供 服务 时 所 需 的 算法 。 因 为 
list0 接 受 FilenameFilter 对 象 作为 参数 ， 这 意味 着 我 们 可 以 传递 实现 了 FilenameFilter 接 口 的 任何 
类 的 对 象 ， 用 以 选择 (甚至 在 运行 时 ) list0 方 法 的 行为 方式 。 策 略 的 目的 就 是 提供 了 代码 行为 
的 灵活 性 。 

accept0 方 法 必须 接受 一 个 代表 某 个 特定 文件 所 在 目录 的 File 对 象 ， 以 及 包含 了 那个 文件 名 
的 一 个 String。 记 住 一 点 : list0 方 法 会 为 此 目录 对 象 下 的 每 个 文件 名 调用 accept0， 来 判断 该 文 
件 是 否 包含 在 内 ， 判断 结果 由 accept0 返 回 的 布尔 值 表示 。 

accept0 会 使 用 一 个 正则 表达 式 的 matcher 对 象 ， 来 查看 此 正则 表达 式 regex 是 否 匹配 这 个 文 
件 的 名 字 。 通 过 使 用 acceptO，list( 方 法 会 返回 一 个 数组 。 

匿名 内 部 类 

这 个 例子 很 适合 用 一 个 匿名 内 部 类 (第 8 章 介绍 过 ) 进行 改写 。 首 先 创建 一 个 fllter0 方 法 ， 
它 会 返回 一 个 指向 FilenameFilter 的 引用 : 


/11: io/DirList2.java 

1/ Uses anonymous inner classes. 
/1 (args: “D.*\.java") 

‘import java.util.regex.*; 
import java.io.*; 
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import java.util.*; 


public class DirList2 { 
public static FilenameFilter filter(final String regex) { 
// Creation of anonymous inner class: 
return new FilenameFilter() { 
private Pattern pattern = Pattern.compile(regex); 
public boolean accept (File dir, String name) { 
return pattern.matcher (name) „matches () ; 
} 
}; // End of anonymous inner class 
中 
public static void main(String[] args) { 
File path = new File(".” 
String[] list; 
if(args.length == 0) 
list = path. list(); 
else 
list = path. list (filter (args(01)); 
Arrays.sort(list, String.CASE_INSENSITIVE_ORDER) ; 
for(String dirItem : list) 
System.out.printin(dirItem) ; 





} 
} /* Output: 
DirectoryDemo. java 
Dirlist.java 
Dirlist2. java 
‘Dirlist3.java 
Wh 


注意 ， 传 向 人 lter0 的 参数 必须 是 final 的 。 这 在 匿名 内 部 类 中 是 必需 的 ， 这 样 它 才能 够 使 用 来 


自 该 类 范围 之 外 的 对 象 。 
这 个 设计 有 所 改进 ， 因 为 现在 FilenameFilter 类 紧密 地 和 DirList2 绑 定 在 一 起 。 然 而 ， 我 们 
可 以 进一步 修改 该 方法 ， 定 义 一 个 作为 list0 和 参数 的 匿名 内 部 类 ， 这 样 一 来 程序 会 变 得 更 小 : 


1/: fo/DirList3.java 
// Building the anonymous inner class "in-place." 
17 (args: "D.*\. java") 

import java.util. regex.*; 

import java.io.*; 

import java.util.*; 





public class Dirlist3 { 
public static void main(final String[] args) { 
File path = new File("."); 
Stringi] list; 
if (args.length == 0) 
list = path.list(); 
else 
list = path. list (new FilenameFilter() { 
private Pattern pattern = Pattern.compile(args(®]): 
public boolean accept(File dir, String name) { 
return pattern.matcher (name) .matches() ; 
y 


H: 
Arrays.sort(list, String.CASE_INSENSITIVE_ORDER) ; 


for(String dirItem : list) 
System.out.println(dirItem); 





2 
} /* Output: 
DirectoryDemo. java 
DirList. java 
1 DirList2.java 
DirList3.java 
AL 
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既然 匿名 内 部 类 直接 使 用 args[0]， 那 么 传递 给 main0 方 法 的 参数 现在 就 是 final 的 。 

这 个 例子 展示 了 匿名 内 部 类 怎样 通过 创建 特定 的 、 一 次 性 的 类 来 解决 问题 。 此 方法 的 一 个 
优点 就 是 将 解决 特定 问题 的 代码 隔离 、 聚 拢 于 一 点 。 而 另 一 方面 ， 这 种 方法 却 不 易 阅读 ， 因 此 
要 谨慎 使 用 。 

练习 1: (3) 修改 DirListjava (或 其 变 体 之 一 )， 以 便 FilenameFilter 能 够 打开 每 个 文件 (使 
用 net.mindview.util.TextFile 工 具 )， 并 检查 命令 行 尾随 的 参数 是 否 存 在 于 那个 文件 中 ， 以 此 检查 
结果 来 决定 是 否 接受 这 个 文件 。 

练习 2: (2) 创建 一 个 叫做 SortedDirList 的 类 ， 它 具有 一 个 可 以 接受 文件 路 径 信息 ， 并 能 构 
建 该 路 径 下 所 有 文件 的 排序 目录 列表 的 构造 器 。 向 这 个 类 添加 两 个 重 载 的 ist0 方 法 : 一 个 产生 
整个 列表 ， 另 一 个 产生 与 其 参数 (一 个 正则 表达 式 ) 相 匹配 的 列表 的 子 集 。 

练习 3: (3) 修改 DirListjava (或 其 变 体 之 一 ) ， 使 其 对 所 选中 的 文件 计算 文件 尺寸 的 总 和 。 
18.1.2 目录 实用 工具 

程序 设计 中 一 项 常见 的 任务 就 是 在 文件 集 上 执行 操作 ， 这 些 文件 要 么 在 本 地 目录 中 ， 要 么 
遍布 于 整个 目录 树 中 。 如 果 有 一 种 工具 能 够 为 你 产生 这 个 文件 集 ， 那 么 它 会 非常 有 用 。 下 面 的 
实用 工具 类 就 可 以 通过 使 用 local() 方 法 产生 由 本 地 目录 中 的 文件 构成 的 File 对 象 数组 ， 或 者 通过 
使 用 walk0 方 法 产生 给 定 目录 下 的 由 整个 目录 树 中 所 有 文件 构成 的 List<File> (File 对 象 比 文件 名 
更 有 用 ， 因 为 File 对 象 包含 更 多 的 信息 )。 这 些 文件 是 基于 你 提供 的 正则 表达 式 而 被 选中 的 ， 


/1/: net/mindview/util/Directory. java 

// Produce a sequence of File objects that match a 
// regular expression in either a local directory, 
// or by walking a directory tree. 

package net .mindview.util; 

import java.util.regex.*; 

import java. io.*; 

import java.util.*; 


public final class Directory { 
public static Fite[] 
local(File dir, final String regex) { 
return dir.listFiles(new FilenameFilter() { 
private Pattern pattern = Pattern. compile(regex); 
Public boolean accept(File dir, String name) { 
return pattern.matcher ( 
new File(name) .getName()).matches() ; 
} 


D: 

} 

public static File[) 

local (String path, final String regex) { // Overloaded 
return local(new File(path), regex); 

} 

// A two-tuple for returning a pair of objects: 

public static class TreeInfo implements Iterable<File> { 
public List<File> files = new ArrayList<File>(); 
public List<File> dirs = new ArrayList<File>(): 
// The default iterable element is the file list: 
public Iterator<File> iterator() { 

return files. iterator(); 


} 

void addAll(TreeInfo other) { 
files.addAll (other. files); 
dirs.addAll(other.dirs); 


} 
Public String toString() { 
return “dirs: ”+ PPrint.pformat(dirs) + 
"\n\nfiles: ”+ PPrint.pformat(files): 
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} 

public static TreeInfo 

walk(String start, String regex) { // Begin recursion 
return recurseDirs(new File(start), regex); 

和 

public static TreeInfo 

walk(File start, String regex) { // Overloaded 
return recurseDirs(start, regex); 


} 
public static Treelnfo walk(File start) { // Everything 
return recurseDirs(start, ".*"); 
} 
public static TreeInfo walk(String start) { 
return recurseDirs(new File(start), ".*"); 
} 
static Treelnfo recurseDirs(File startDir, String regex){ 
Treelnfo result = new TreeInfo(): 
for(File item : startDir.listFiles()) { 
if(item.isDirectory()) { 
result.dirs.add(item) ; 
result.addAll(recurseDirs(item, regex); 
} else // Regular file 
if (item. getName() .matches(regex)) 
result. files.add(1tem) ; 


return result; 
i Simple validation test: 
public static void main(String] args) { 
if(args. length == 0) 
System.out.printin(walk(".")); 
else 
for (String arg : args) 
System.out.printin(walk(arg)); 
} his s 
local SiR MERI tE e HA ROHR EMER. 7 可 以 看 到 ， 它 还 使 用 了 
FilenameFilter。 如 果 需 要 List 而 不 是 数组 ， 你 可 以 使 用 Arrays.asList0 自 己 对 结果 进行 转换 。 
walk() 方 法 将 开始 目录 的 名 字 转 换 为 File 对 象 ， 然 后 调用 recurseDirs0， 该 方法 将 递归 地 遍 
历 目录 ， 并 在 每 次 递归 中 都 收集 更 多 的 信息 。 为 了 区 分 普通 文件 和 目录 ， 返 回 值 实际 上 是 一 个 
对 象 “元 组 ”一 一 个 List 持 有 所 有 普通 文件 ， 另 一 个 持 有 目录 。 这 里 ， 所 有 的 域 都 被 有 意识 地 
设置 成 了 public， 因 为 TreeInfo 的 使 命 只 是 将 对 象 收集 起 来 一 一 如 果 你 只 是 返回 List， 那 么 就 不 
需要 将 其 设置 为 private， 因 为 你 只 是 返回 一 个 对 象 对 ， 不 需要 将 它们 设置 为 private。 注 意 ， 
TreeInfo 实 现 了 Iterable<File>， 它 将 产生 文件 ， 使 你 拥有 在 文件 列表 上 的 “默认 磷 代 ”"， 而 你 可 
以 通过 声明 “.dirs” 来 指定 目录 。 
TreelInfo.toString0 方 法 使 用 了 一 个 “灵巧 打印 机 ”类 ， 以 使 输出 更 容易 浏览 。 容 器 默认 的 
toString() 方 法 会 在 单个 行 中 打印 容器 中 的 所 有 元 素 ， 对 于 大 型 集合 来 说 ， 这 会 变 得 难以 阅读 ， 
因此 你 可 能 希望 使 用 可 替换 的 格式 化 机 制 。 下 面 是 一 个 可 以 添加 新 行 并 缩 排 所 有 元 素 的 工具 ， 


//: net/mindview/util/PPrint. java 

// Pretty-printer for collections 

package net.mindview.util; 

import java.util.*; 

public class PPrint { 

public static String pformat(Collection<?> c) { 

if(c.size() == 0) return “[]": 
StringBuilder result = new StringBuilder ("["); 
for (Object elem : c) { 
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if(c.sizeQ) != 1) 
result.append("\n "); 
result. append(elem) ; 


J 
if(c.size() != 1) 

result .append("\n") ; 
result. append("}"); 
return result. toString(); 


} 
public static void pprint(Collection<?> c) { 
System. out.printIn(pformat(c)): 


} 
public static void pprint(Object[] c) { 
System. out. printin(pformat (Arrays.asList(c))); 


} 

} Ii~ 

pformat0 方 法 可 以 从 Collection 中 产生 格式 化 的 String， 而 pprint0 方 法 使 用 pformat0 来 执行 
其 任务 。 注 意 ， 没 有 任何 元 素 和 只 有 一 个 元 素 这 两 种 特例 进行 了 不 同 的 处 理 。 上 面 还 有 一 个 用 
于 数组 的 pprint0 版 本 。 

Directory 实 用 工具 放 在 了 net.mindview.util 包 中 ， 以 使 其 可 以 更 容易 地 被 获得 。 下 面 的 例子 
说 明了 你 可 以 如 何 使 用 它 的 样本 : 


//: i0/DirectoryDemo. java 
// Sample use of Directory utilities. 
import java.io.*; 

import net.mindview.util.*; 

import static net.mindview.util.Print.*; 


public class DirectoryDemo { 
public static void main(Stringl] args) { 
1/ ALL directories: 
PPrint.pprint (Directory.walk(".").dirs); 
// MI files beginning with 'T' 
for(File file : Directory.locat(".", "T.*")) 
print(file); 
print" ji 
// ALL Java files beginning with 'T': 
for(File file : Directory.watk(".", "T.*\\.java")) 
print(file); 
print("====================: : 
// Class files containing "Z" or 
for(File file : Directory.walk(" 
print (file): 









(Zz] .*\\.class")) 


} 
} /* Output: (Sample) 
[.\xfiles] 
,ATestEOF .class 
+\TestEOF .java 
‘\TransferTo.class 
.\TransferTo. java 
.ATestEOF ,java 
-\TransferTo. java 
-\xfiles\ThawAlien. java 








« \FreezeAlien.c ass 
-\GZIPcompress.class 

«\ZipCompress.class 

H~ 

你 可 能 需要 更 新 一 下 在 第 13 章 中 学 习 到 的 有 关 正 则 表达 式 的 知识 ， 以 理解 在 local0 和 walkO 
二 个 参数 。 

我 们 可 以 更 进一步 ， 创 建 一 个 工具 ， 它 可 以 在 目录 中 穿行 ， 并 且 根 据 Strategy 对 象 来 处 理 这 


中 的 
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| 些 目录 中 的 文件 (这 是 策略 设计 模式 的 另 一 个 示例 ): 


//: net/mindview/util/ProcessFiles. java 
package net.mindview.util; 
import java.io.*; 





public class ProcessFiles { 
public interface Strategy { 
void process(File file); 
} 
private Strategy strategy; 
private String ext; 
public ProcessFiles(Strategy strategy, String ext) { 
this.strategy = strategy; 
this.ext = ext; 
} 
public void start(String{] args) { 
try { 
if (args.length == 0) 
processDirectoryTree(new File(".")); 
else 
for(String arg : args) { 
File fileArg = new File(arg); 
if (fileArg.{sDirectory()) 
processDirectoryTree(fileArg) ; 
else { 
// Allow user to Leave off extension: 
if(larg.endswith("." + ext)) 
arg += "." + ext; 
strategy. process ( 
new File(arg) .getCanonicatFite()); 
} 


)》 
} catch(IOException e) { 
throw new RuntimeException(e) ; 


} 


} 
public void 
ProcessDirectoryTree(File root) throws IOException { 
for(File file : Directory.walk( 
root.getAbsolutePath(), *.*\\." + ext)) 
strategy. process(file.getCanonicalFile()); 


} 
// Demonstration of how to use it: 
public static void main(String(] args) { 
new ProcessFiles(new ProcessFiles.Strategy() { 
Public void process(File file) { 
System.out.printin(file); 


2} 
}. "java").start(args); 
) /* (Execute to see output) *///:~ 


Strategy 接 口内 嵌 在 ProcessFiles 中 ， 使 得 如 果 你 希望 实现 它 ， 就 必须 实现 ProcessFiles. 
Strategy， 它 为 读者 提供 了 更 多 的 上 下 文 信息 。ProcessFiles 执 行 了 查找 具有 特定 扩展 名 (传递 
给 构造 器 的 ext 参 数 ) 的 文件 所 需 的 全 部 工作 ， 并 且 当 它 找到 匹配 的 文件 时 ， 将 直接 把 文件 传递 
给 Strategy 对 象 (也 是 传递 给 构造 器 的 参数 ) 。 

如 果 你 没有 提供 任何 参数 ， 那 么 ProcessFiles 就 假设 你 希望 遍历 当前 目录 下 的 所 有 目录 。 你 
也 可 以 指定 特定 的 文件 ， 带 不 带 扩展 名 都 可 以 (如 果 必 和 需 的 话 ， 它 会 添加 上 扩展 名 )， 或 者 指定 
一 个 或 多 个 目录 。 

| 在 main0 中 ， 你 看 到 了 如 何 使 用 这 个 工具 的 基本 示例 ， 它 可 以 根据 你 提供 的 命令 行 来 打印 

所 有 的 Java 源 代码 文件 的 名 字 。 
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练习 4: (2) 使 用 Directory.walk0 来 计算 在 目录 中 所 有 名 字 与 特定 的 正则 表达 式 相 匹配 的 文 
件 的 尺寸 总 和 。 
练习 5: (1) 修改 ProcessFiles,java， 使 其 匹配 正则 表达 式 而 不 是 固定 的 扩展 名 。 


18.1.3 目录 的 检查 及 创建 

File 类 不 仅仅 只 代表 存在 的 文件 或 目录 。 也 可 以 用 File 对 象 来 创建 新 的 目录 或 尚 不 存在 的 整 
个 目录 路 径 。 我 们 还 可 以 查看 文件 的 特性 (如: 大 小 ， 最 后 修改 日 期 读 / 写 ) ， 检 查 某 个 File 对 
象 代表 的 是 一 个 文件 还 是 一 个 目录 ， 并 可 以 删除 文件 。 下 面 的 示例 展示 了 File 类 的 一 些 其 他 方法 
(请 参考 http://java.sun.com 上 的 HTML 文 档 以 全 面 了 解 它们 )。 


//: io/MakeDirectories.java 
// Demonstrates the use of the File class to 
// Create directories and manipulate files. 

// {Args: MakeDirectoriesTest} 

import java.io.*; 


public class MakeDirectories { 
private static void usage() { 
System.err.printin( 
“Usage: MakeDirectories pathl 
"Creates each path\n" + 
“Usage: MakeDirectortes -d pathl ...\n" + 
“Deletes each path\n” + 
“Usage :MakeDirectories -r path] path2\n" + 
"Renames from pathl to path2"); 
System.exit(1); 
r 
Private static void fileData(File f) { 
System. out.printin( 
“Absolute path: " + f.getAbsolutePath() + 
"\n Can read + f.canRead() + 
"\n Can write: " + f.canWrite() + 
"\n getName: ”+ f.getName() + 
+ 


\n" + 








"\n getParent: " + f.getParent() + 

"\n getPath: ”+ f.getPath() + 

"\n Length: " + f.length() + 

"\n lastModified: ”+ f.lastModified()): 
if (f.isFile()) 

System.out.printin("It's a file"); 
else if(f.isDirectory()) 

System.out.printin("It's a directory"); 








} 
public static void main(Stringl] args) { 
if(args.length < 1) usage(); 
if (args[0] .equals("-r")) { 
if(args.length != 3) usage(); 
File 
old = new File(args{1]), 
rname = new File(args{2]); 
old. renameTo(rname) ; 
fileData(old); 
fileData(rname) ; 
return; // Exit main 





} 

int count = 6; 

boolean del = false; 

if (args[0] .equats(*-d")) { 


count++; 
del = true; 

} 

count--; 


while(+tcount < args.length) { 
File f = new File(args{count]); 
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if(f.exists()) { 
System.out.printIn(f + " exists"); 
if (det) { 
System. out.printIn("deleting..." + f); 
f.delete(): 
} 
} 
else { // Doesn't exist 
if(idel) { 
f mkdirs(); 
System.out.printin("created " + f); 


y 
} 
fileData(f); 


} 
} /* Output: (80% match) 
created MakeDirectoriesTest 
Absolute path: d:\aaa-TIJ4\code\io\MakeDirectoriesTest 
Can read: true 

Can write: true 

getName: MakeDirectoriesTest 
getParent: null 

getPath: MakeDirectoriesTest 
length: @ 

lastModified: 1101690308831 
It's a directory 
“Ii~ 


在 fileData0 中 ， 可 以 看 到 用 到 了 多 种 不 同 的 文件 特征 查询 方法 来 显示 文件 或 目录 路 径 的 
信息 。main() 方 法 首先 调用 的 是 renameTo0 ， 用 来 把 一 个 文件 重 命名 (或 移动 ) 到 由 参数 所 
指示 的 另 一 个 完全 不 同 的 新 路 径 〈 也 就 是 另 一 个 File 对 象 ) 下 面 。 这 同样 适用 于 任意 长 度 的 文 
件 目录 。 

实践 上 面 的 程序 可 以 发 现 ， 我 们 可 以 产生 任意 复杂 的 目录 路 径 ， 因 为 mkdirs0 可 以 为 我 们 做 
好 这 一 切 。 

练习 6: (5) 使 用 ProcessFiles 来 查找 在 某 个 特定 目录 子 树 下 的 所 有 在 某 个 特定 日 期 之 后 进行 
过 修改 的 Java 源 代码 文件 。 


18.2 输入 和 输出 


编程 语言 的 WO 类 库 中 常 使 用 流 这 个 抽象 概念 ， 它 代表 任何 有 能 力 产 出 数据 的 数据 源 对 象 或 
者 是 有 能 力 接收 数据 的 接收 端 对 象 。“ 流 ”屏蔽 了 实际 的 VO 设备 中 处 理 数据 的 细节 。 

Java 类 库 中 的 MO 类 分 成 输入 和 输出 两 部 分 ， 可 以 在 JDK 文 档 里 的 类 层次 结构 中 查看 到 。 通 
过 继承 ， 任 何 自 Inputstream 或 Reader 派 生 而 来 的 类 都 含有 名 为 read0 的 基本 方法 ， 用 于 读 取 单 
个 字 节 或 者 字 节 数组 。 同 样 ， 任 何 自 OutputStream 或 Writer 派 生 而 来 的 类 都 含有 名 为 write0 的 
基本 方法 ， 用 于 写 单个 字 节 或 者 字 节 数组 。 但 是 ， 我 们 通常 不 会 用 到 这 些 方法 ， 它 们 之 所 以 存 
在 是 因为 别 的 类 可 以 使 用 它们 ， 以 便 提供 更 有 用 的 接口 。 因 此 ， 我 们 很 少 使 用 单一 的 类 来 创建 
流 对 象 ， 而 是 通过 又 合 多 个 对 象 来 提供 所 期 望 的 功能 (这 是 装饰 器 设计 模式 ， 你 将 在 本 节 中 看 
到 它 )。 实 际 上 ，Java 中 “ 流 ” 类 库 让 人 迷惑 的 主要 原因 就 在 于 : 创建 单一 的 结果 流 ， 却 需要 创 
建 多 个 对 象 。 

有 必要 按照 这 些 类 的 功能 对 它们 进行 分 类 。 在 Java 1.0 中 ， 类 库 的 设计 者 首先 限定 与 输 
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人 和 人 有关 的 所 有 类 都 应 该 从 InputStream 继 承 ， 而 与 输出 有 关 的 所 有 类 都 应 该 从 OutputStream 


继承 。 


正如 在 本 书 中 所 实践 的 ， 我 将 尝试 着 提供 这 些 类 的 总 找 ， 但 是 我 必须 假设 你 确实 将 会 使 用 
JDK 文 档 来 确定 所 有 的 细节 ， 例 如 某 个 特定 类 的 详尽 的 方法 列表 。 
18.2.1 InputStream 类 型 
JInputStream 的 作用 是 用 来 表示 那些 从 不 同 数据 源 产 生 输 入 的 类 。 如 表 18-1 所 示 ， 这 些 数据 


源 包括 : 
D 字 节 数组 。 
2) String 对 象 。 
3) 文件 。 


4) “管道 "， 工 作 方式 与 实际 管道 相似 ， 即 ， 从 一 端 输 入 ， 从 另 一 端 输出 。 
5) 一 个 由 其 他 种 类 的 流 组 成 的 序列 ， 以 便 我 们 可 以 将 它们 收集 合并 到 一 个 流 内 。 
6) 其 他 数据 源 ， 如 Internet 连 接 等 (参见 可 以 在 www.MindView.net 获 得 的 《Thinking in 


Enterprise Java))) 。 


每 一 种 数据 源 都 有 相应 的 InputStream 子 类 。 另 外 , FilterfnputStream 也 属于 一 种 InputStream , 
为 “装饰 器 ”(decorator) 类 提供 基 类 ， 其 中 ,“ 装 饰 器 ”类 可 以 把 属性 或 有 用 的 接口 与 输入 流连 
接 在 一 起 。 我 们 稍 后 再 讨论 它 。 


表 18-1 InputStream 类 型 


























类 功 能 构造 器 参数 
如 何 使 用 
ByteArrayInputStream 无 许 将 内 存 的 缓冲 区 当 作 InputStream | ”缓冲 区 ， 字 节 将 从 中 取出 
使 用 作为 一 种 数据 源 ， 将 其 与 FilterInputStream 
对 象 相连 以 提供 有 用 接口 
StringBufferInputStream 将 String 转 换 成 InputStream 字符 串 。 底 县 实现 实际 使 用 StringBuffer 
作为 一 种 数据 源 : 将 其 与 FilterInputStream 
对 象 相连 以 提供 有 用 接口 
FileInputStream 用 于 从 文件 中 读 取 信息 字符 串 ， 表 示 文 件 名 、 文 件 或 FileDeseriptor 
对 象 
作为 一 种 数据 源 : 将 其 与 FilterInputStream 
对 象 相连 以 提供 有 用 接口 
PipedInputStream 产生 用 于 写 人 相关 PipedOutput Stream | PipedOutputStream 
的 数据 。 实 现 “管道 化 ”概念 作为 多 线程 中 数据 源 : 将 其 与 FilterInput- 
Stream 对 象 相 连 以 提供 有 用 接口 
SequenceInputStream 将 隔 个 或 多 个 InputStream 对 象 转换 成 | 两 个 InputStream 对 象 或 一 个 容纳 inputStream 
单一 InputStream 对 象 的 容器 Enumeration 
作为 一 种 数据 源 : 将 其 与 FilterInputStream 
对 象 相连 以 提供 有 用 接口 
FitterInputStream 抽象 类 ， 作 为 “装饰 器" 的 接口 。 其 中 ，| mz 18-3 
“装饰 器 ”为 其 他 的 InputStream 类 提供 | 见 表 18-3 
有 用 功能 。 见 表 18-3 
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18.2.2 OutputStream 类 型 


如 表 18-2 所 示 ， 该 类 别 的 类 决定 了 输出 所 要 去 往 的 目标 : 字 节 数组 (但 不 是 String， 不 过 你 
当然 可 以 用 字 节 数组 自己 创建 )、 文 件 或 管道 。 
另外 ，FilterOutputStream 为 “装饰 器 ”类 提供 了 一 个 基 类 ,“ 装 饰 器 ”类 把 属性 或 者 有 用 
的 接口 与 输出 流连 接 了 起 来 ， 这 些 稍 后 会 讨论 。 


类 


表 18-2 OutputStream 类 型 





功 能 


构造 器 参数 
如 何 使 用 





ByteArrayOutputStream 


在 内 存 中 创建 缓冲 区 。 所 有 送 往 “ 流 ” 


-的 数据 都 要 放置 在 此 缓冲 区 


缓冲 区 初始 化 尺寸 (可 选 的 ) 
用 于 指定 数据 的 目的 地 将 其 与 Filter- 
OutputStream 对 象 相连 以 提供 有 用 接口 





FileOutputStream 


用 于 将 信息 写 至 文件 


字符 串 ， 表 示 文 件 名 、 文 件 或 File- 
Descriptor 对 象 

指定 数据 的 目的 地 ， 将 其 与 Filter- 
OutputStream 对 象 相连 以 提供 有 用 接口 





PipedOutputStream 


任何 写 人 其 中 的 信息 都 会 自动 作为 相 
关 PipedInputStream 的 给 出 。 实 现 “ 管 
道 化 ”概念 。 


PipedInputStream 

指定 用 于 多 线程 的 数据 的 目的 地 ， 将 其 
与 FilterOutputStream 对 象 相连 以 提供 有 
用 接口 





FilterOutputStream 





抽象 类 ， 作 为 “装饰 器 ”的 接口 。 其 中 ， 
“装饰 器 ”为 其 他 OutputStream 提供 有 
用 功能 。 见 表 18-4 





见 表 18-4 
见 表 18-4 





18.3 添加 属性 和 有 用 的 接口 


装饰 器 在 第 15 章 引入 。Java IO 类 库 需 要 多 种 不 同 功能 的 组 合 ， 这 正 是 使 用 装饰 器 模式 的 理 
由 所 在 ” 。 这 也 是 Java IO 类 库 里 存在 fiter (过 滤器 ) 类 的 原因 所 在 抽象 类 全 ter 是 所 有 装饰 器 类 
的 基 类 。 装 饰 器 必须 具有 和 它 所 装饰 的 对 象 相 同 的 接口 ， 但 它 也 可 以 扩展 接口 ， 而 这 种 情况 只 


发 生 在 个 别 filter 类 中 。 


但 是 ， 装 饰 器 模式 也 有 一 个 缺点 ;在 编写 程序 时 ， 它 给 我 们 提供 了 相当 多 的 灵活 性 (因为 
我 们 可 以 很 容易 地 混合 和 匹配 属性 ) ， 但 是 它 同 时 也 增加 了 代码 的 复杂 性 。Java IO 类 库 操作 不 
便 的 原因 在 于 :我 们 必须 创建 许多 类 一 一 “核心 ”1/O 类 型 加 上 所 有 的 装饰 器 ， 才 能 得 到 我 们 所 


希望 的 单个 1O 对 象 。 


FilterInputStream 和 FilterOutputStream 是 用 来 提供 装饰 器 类 接口 以 控制 特定 输入 流 
(InputStream) 和 输出 流 (OutputStream) 的 两 个 类 ， 它 们 的 名 字 并 不 是 很 直观 。FilterInput- 
Stream 和 FilterOutputStream 分 别 自 IJO 类 库 中 的 基 类 InputStream 和 OutputStream 派 生 而 来 ， 这 
两 个 类 是 装饰 器 的 必要 条 件 (以便 能 为 所 有 正在 被 修饰 的 对 象 提供 通用 接口 )。 

18.3.1 通过 FilterInputStream 从 InputStream 读 取 数 据 

FilterInputStream 类 能 够 完成 两 件 完全 不 同 的 事情 。 其 中 ，DataInputStream 人 允许 我 们 读 取 


O 很 难说 这 就 是 一 个 很 好 的 设计 选择 ， 尤 其 是 与 其 他 程序 设计 语言 中 的 简单 VO 类 库 相 比较 。 但 它 的 确 是 如 此 选 


择 的 一 个 恰当 理由 。 
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不 同 的 基本 类 型 数据 以 及 String 对 象 (所 有 方法 都 以 “read” 开 头 ， 例 如 readByte0、readFloat0 
等 等 )。 搭 配 相应 的 DataOutputStream， 我 们 就 可 以 通过 数据 “ 流 ” 将 基本 类 型 的 数据 从 一 个 
地 方 迁移 到 另 一 个 地 方 。 具 体 是 哪些 “地 方 ”是 由 表 18-1 中 的 那些 类 决定 的 。 

其 他 FilterInputstream 类 则 在 内 部 修改 InputStream 的 行为 方式 ， 是 否 缓冲 ， 是 否 保留 它 所 
读 过 的 行 〔 允 许 我 们 查询 行 数 或 设置 行 数 )， 以 及 是 否 把 单一 字符 推 回 输入 流 等 等 。 最 后 两 个 类 
看 起 来 更 像 是 为 了 创建 一 个 编译 器 (它们 被 添加 进来 可 能 是 为 了 对 “用 Java 构 建 编译 器 ”实验 
提供 支持 )， 因 此 我 们 在 一 般 编程 中 不 会 用 到 它们 。 

我 们 几乎 每 次 都 要 对 输入 进行 缓冲 一 一 不 管 我 们 正在 连接 的 是 什么 WO 设备 ， 所 以 ，1/O 类 库 
把 无 缓冲 输入 (而 不 是 缓冲 输入 ) 作为 特殊 情况 (或 只 是 方法 调用 ) 就 显得 更 加 合理 了 。 
FilterInputStream 的 类 型 及 功能 如 表 18-3 所 示 。 


表 18-3 FilterinputStream 类 型 


构造 器 参数 


类 功 能 
如 何 使 用 





DataInputStream 与 DataOutputStream 搭 配 使 用 ， 因 此 InputStream 
我 们 可 以 按照 可 移植 方式 从 流 读 取 基 本 包含 用 于 读 取 基 本 类 型 数据 的 全 部 接口 
数据 类 型 int char, long 等 ) 
BufferedInputStream 使 用 它 可 以 防止 每 次 读 取 时 都 得 进行 实 JInputStream， 可 以 指定 级 冲 区 大 小 (可 
际 写 操作 。 代 表 “ 使 用 缓冲 区 ” 选 的 ) 

本 质 上 不 提供 接口 ， 只 不 过 是 向 进程 中 


添加 缓冲 区 所 必需 的 。 与 接口 对 象 搭配 
LineNumberInputStream 跟踪 输入 流 中 的 行 号 ， 可 调用 getLine InputStream 











Number() 和 setLineNumber(int) 仅 增加 了 行 号 ， 因 此 可 能 要 与 接口 对 象 
搭配 使 用 
PushbackInputStream 具有 “能 弹出 一 个 字 节 的 缓冲 区 "。 因 | InputStream 
此 可 以 将 读 到 的 最 后 一 个 字符 回 退 通常 作为 编译 器 的 扫描 器 ， 之 所 以 包含 


在 内 是 因为 Java 编 译 器 的 需要 ， 我 们 可 能 
永远 不 会 用 到 
18.3.2 通过 FilterOutPutStream 向 OutputStream 写 入 

与 DataInputStream 对 应 的 是 DataOutputStream， 它 可 以 将 各 种 基本 数据 类 型 以 及 String 对 
象 格式 化 输出 到 “ 流 ” 中 ， 这 样 以 来 ， 任 何 机 器 上 的 任何 DataInputStream 都 能 够 读 取 它 们 。 所 
有 方法 都 以 “wirte” 开 头 ， 例 如 writeByte0、writeFloat0 等 等 。 

PrintStream 最 初 的 目的 便 是 为 了 以 可 视 化 格式 打印 所 有 的 基本 数据 类 型 以 及 String 对 象 。 
这 和 DataOutputStream 不 同 ， 后 者 的 目的 是 将 数据 元 素 置 人“ 流 ”中 ， 使 DataInputStream 能 
够 可 移植 地 重 构 它们 。 

PrintStream 内 有 两 个 重要 的 方法 : print0 和 printin0。 对 它们 进行 了 重 载 ， 以 便 可 打印 出 
各 种 数据 类 型 。print0 和 printin0 之 间 的 差异 是 ， 后 者 在 操作 完毕 后 会 添加 一 个 换行 符 。 

了 PrintStream 可 能 会 有 些 问题 ， 因 为 它 捕 所 了 所 有 的 IOExceptions 因此， 我们 必须 使 用 
checkError0 自 行 测试 错误 状态 ,如 果 出 现 错误 它 返 回 true)。 另 外 ，PrintStream 也 未 完全 国际 化 ， 
不 能 以 平台 无 关 的 方式 处 理 换行 动作 (这些 问题 在 printWriter 中 得 到 了 解决 ， 这 在 后 面 讲述 )。 

BufferedOutputStream 是 一 个 修改 过 的 OutputStream， 它 对 数据 流 使 用 缓冲 技术 ， 因 此 当 
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每 次 向 流 写 人 时 ， 不 必 每 次 都 进行 实际 的 物理 写 动作 。 所 以 在 进行 输出 时 ， 我 们 可 能 更 经 常 的 
是 使 用 它 。FilterOutputStream 的 类 型 及 功能 如 表 18-4 所 示 。 


表 18-4 FilterOutputStream 类 型 








构造 器 参数 
类 功 能 
如 何 使 用 
DataOutputStream 与 DataInputStream 搭配 使 用 ， 因 此 OutputStream 


可 以 按照 可 移植 方式 向 流 中 写 人 基本 类 包含 用 于 写 人 基本 类 型 数据 的 全 部 接口 
型 数据 (int char, long 等 ) 





PrintStream 用 于 产生 格式 化 输出 。 其 中 DataOut。 | 。 OutputStream, 可 以 用 boolean 值 指示 是 
putStream 处 理 数据 的 存储 ，PrintStream | 否 在 每 次 换行 时 清空 缓冲 区 〈 可 选 的 ) 应 
处 理 显示 | 该 是 对 Outputstream 对 象 的 “final” 封装 。 

可 能 会 经 常 使 用 到 它 





BufferedOutputStream 使 用 它 以 避免 每 次 发 送 数据 时 都 要 进 OutputStream, 可 以 指定 缓冲 区 大 小 〈 可 
行 实际 的 写 操作 。 代 表 “ 使 用 缓冲 区 "。 | 选 的 ) 

可 以 调用 flush0 清空 缓冲 区 本 质 上 并 不 提供 接口 ， 只 不 过 是 向 进程 
中 添加 缓冲 区 所 必需 的 。 与 接口 对 象 搭配 








18.4 _ Reader 和 Writer 


Java 1.1 对 基本 的 VO 流 类 库 进行 了 重大 的 修改 。 当 我 们 初次 看 见 Reader 和 Writer 类 时 ， 可 能 
会 以 为 这 是 两 个 用 来 替代 InputStream 和 OutputStreamt 的 类 ， 但 实际 上 并 非 如 此 。 尽 管 一 些 原 
始 的 “ 流 ”类 库 不 再 被 使 用 〈 如 果 使 用 它们 ， 则 会 收 到 编译 器 的 警告 信息 ) ， 但 是 InputStream 
和 OutputStreamt 在 以 面向 字 节 形式 的 MO 中 仍 可 以 提供 极 有 价值 的 功能 ，Reader 和 Writer 则 提 
供 兼容 Unicode 与 面向 字符 的 VO 功能 。 另 外 : 

1) Java 1.1 向 InputStream 和 OutputStreamt 继 承 层次 结构 中 添加 了 一 些 新 类 ， 所 以 显然 这 两 
个 类 是 不 会 被 取代 的 。 

2) 有 时 我 们 必须 把 来 自 于 “ 字 节 ”层次 结构 中 的 类 和 “字符 ”层次 结构 中 的 类 结合 起 来 使 
用 。 为 了 实现 这 个 目的 ， 要 用 到 “适配器 ”(adapter) 类 : InputStreamReader 可 以 把 
InputStream 转 换 为 Reader， 而 OutputStream Writer 可 以 把 OutputStream 转 换 为 Writer。 

设计 Reader 和 Writer 继 承 层次 结构 主要 是 为 了 国际 化 。 老 的 1/O 流 继承 层次 结构 仅 支持 8 位 
字 节 流 ， 并 且 不 能 很 好 地 处 理 16 位 的 Unicode 字 符 。 由 于 Unicode 用 于 字符 国际 化 《Java 本身 的 
char 也 是 16 位 的 Unicode) ， 所 以 添加 Reader 和 Writer 继 承 层次 结构 就 是 为 了 在 所 有 的 IO 操作 中 
都 支持 Unicode。 另 外 ， 新 类 库 的 设计 使 得 它 的 操作 比 旧 类 库 更 快 。 

一 如 本 书 惯例 ， 我 会 尽力 给 出 所 有 类 的 概观 ， 但 是 我 还 要 假定 你 会 自行 使 用 IDK 文 档 查 看 
细节 ， 例 如 方法 的 详尽 列表 。 

18.4.1 数据 的 来 源 和 去 处 

几乎 所 有 原始 的 Java IO 流 类 都 有 相应 的 Reader 和 Writer 类 来 提供 天 然 的 Unicode 操 作 。 
然而 在 某 些 场合 ， 面 向 字 节 的 InputStream 和 OutputStream 才 是 正确 的 解决 方案 ， 特 别 是 ， 
java.util.zip 类 库 就 是 面向 字 节 的 而 不 是 面向 字符 的 。 因 此 ， 最 明智 的 做 法 是 尽量 尝试 使 用 
Reader 和 Writer， 一 旦 程序 代码 无 法 成 功 编译 ， 我 们 就 会 发 现 自己 不 得 不 使 用 面向 字 节 的 
类 库 。 
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下 面 的 表 展 示 了 在 两 个 继承 层次 结构 中 ， 信 息 的 来 源 和 去 处 〈 即 数据 物理 上 来 自 哪 里 及 去 


ARE) 之 间 的 对 应 关系 : 








来 源 与 去 处 ， Java 1.0 类 相应 的 Java 1.1 类 
InputStream Reader 

适配器 : InputStreamReader 
OutputStream Writer 

适配器 ; OutputStreamWriter 
FileInputStream FileReader 
FileOutputStream Filewriter 
StringBufferInputStream( © fë H) StringReader 
(无 相应 的 类 ) StringWriter 
ByteArrayInputStream CharArrayReader 
ByteArrayOutputStream CharArrayWriter 
PipedInputStream PipedReader 


PipedOutputStream 


PipedWriter 


一 -一 
大 体 上 ， 我 们 会 发 现 ， 这 两 个 不 同 的 继承 层次 结构 中 的 接口 即使 不 能 说 完全 相同 ， 但 也 是 


非常 相似 。 
18.4.2 更 改 流 的 行为 


对 于 InputStream 和 OutputStream 来 说 ， 我 们 会 使 用 FilterInputStream 和 FilterOutputStream 
的 装饰 器 子 类 来 修改 “ 流 ” 以 满足 特殊 需要 。Reader 和 Writer 的 类 继承 层次 结构 继续 沿用 相同 的 


思想 一 -但 是 并 不 完全 相同 。 


在 下 表 中 ， 相 对 于 前 一 表格 来 说 ， 左 右 之 间 的 对 应 关系 的 近似 程度 更 加 粗略 一 些 。 造 成 这 
种 差别 的 原因 是 因为 类 的 组 织 形式 不 同 ， 尽 管 BufferedOutputStream 是 FilterOutputStream 的 子 
类 ,但 是 BufferedWriter 并 不 是 FilterWriter 的 子 类 (尽管 FilterWriter 是 抽象 类 ， 没 有 任何 子 类 ， 
把 它 放 在 那里 也 只 是 把 它 作为 一 个 占 位 符 , 或 仅仅 让 我 们 不 会 对 它 所 在 的 地 方 产生 疑惑 )。 然 而 ， 


这 些 类 的 接口 却 十 分 相似 。 
过 滤器 ，Java 1.0 类 





FilterInputStream 
FilterOutputStream 
BufferedinputStream 
BufferedOutputStream 
DatalnputStream 


PrintStream 
LineNumberInputStream (已 弃 用 ) 
StreamTokenizer 
PushbackInputStream 


相应 的 Java 1.1 类 


FilterReader 
FilterWriter (抽象 类 ， 没 有 子 类 ) 

BufferedReader (也 有 readLine()) 

BufferedWriter 

使 用 DatalnputStream (除了 当 希 要 使 用 readLine0 时 以 外 ， 这 时 应 该 


使 用 BufferedReader) 


PrintWriter 
LineNumberReader 

StreamTokenizer( (使 用 接受 Reader 的 构造 器 ) 
PushbackReader 


有 一 点 很 清楚 : 无 论 我 们 何 时 使 用 readLine0 ， 都 不 应 该 使 用 DataInputStream (这 会 遭 到 


编译 器 的 
的 首选 成 员 。 





列 反 对 ) ， 而 应 该 使 用 BufferedReader。 除 了 这 一 点 ，DataInputStream 仍 是 IO 类 库 


为 了 更 容易 地 过 渡 到 使 用 PrintWriter， 它 提供 了 一 个 既 能 接受 Writer 对 象 又 能 接受 任何 
OutputStream 对 象 的 构造 器 。PrintWriter 的 格式 化 接口 实际 上 与 PrintStream 相 同 。 
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在 Java SE5 中 添加 了 PrintWriter 构 造 器 ， 以 简化 在 将 输出 写 人 时 的 文件 创建 过 程 ， 你 马上 
就 会 看 到 它 。 

有 一 种 PrintWriter 构 造 器 还 有 一 个 选项 ， 就 是 “自动 执行 清空 ”选项 。 如 果 构 造 器 设置 此 
选项 ， 则 在 每 个 Printin0 执 行 之 后 ， 便 会 自动 清空 。 
18.4.3 未 发 生变 化 的 类 

有 一 些 类 在 Java 1.0 和 Java 1.1 之 间 则 未 做 改变 。 


以 下 这 些 Java 1.0 类 在 Java 1.1 中 没有 相应 类 


DataOutputStream 
File 


RandomAccessFile 
SequenceInputStream 


特别 是 DataOutputStream， 在 使 用 时 没有 任何 变化 ， 因 此 如 果 想 以 “可 传输 的 ”格式 存储 
和 检索 数据 ， 可 以 使 用 InputStream 和 OutputStream 继 承 层次 结构 。 


18.5 自我 独立 的 类 : RandomAccessFile > 


RandomAccessFile 适 用 于 由 大 小 已 知 的 记录 组 成 的 文件 ， 所 以 我 们 可 以 使 用 seek0 将 记录 从 
一 处 转移 到 另 一 处 ， 然 后 读 取 或 者 修改 记录 。 文 件 中 记录 的 大 小 不 一 定 都 相同 ， 只 要 我 们 能 够 
确定 那些 记录 有 多 大 以 及 它们 在 文件 中 的 位 置 即 可 。 

最 初 ， 我 们 可 能 难以 相信 RandomAccessFile 不 是 InputStream 或 者 OutputStream 继 承 层次 结 
构 中 的 一 部 分 。 除 了 实现 了 DataInput 和 DataOutput 接 口 (DataInputStream 和 DataOutputStream 
也 实现 了 这 两 个 接口 ) 之 外 ， 它 和 这 两 个 继承 层次 结构 没有 任何 关联 。 它 甚至 不 使 用 
InputStream 和 OutputStream 类 中 已 有 的 任何 功能 。 它 是 一 个 完全 独立 的 类 ， 从 头 开 始 编写 其 所 
有 的 方法 〈 大 多 数 都 是 本 地 的 ) 。 这 么 做 是 因为 RandomAccessFile 拥 有 和 别 的 UO 类 型 本 质 不 同 的 
行为 ， 因 为 我 们 可 以 在 一 个 文件 内 向 前 和 向 后 移动 。 在 任何 情况 下 ， 它 都 是 自我 独立 的 ， 直 接 从 
Object 派生 而 来 。 

从 本 质 上 来 说 ，RandomAccessFile 的 工作 方式 类 似 于 把 DataInputStream 和 DataOutStream 
组 合 起 来 使 用 ， 还 添加 了 一 些 方法 。 共 中 方法 getFilePointer0 用 于 查找 当前 所 处 的 文件 位 置 ， 
seek0 用 于 在 文件 内 移 至 新 的 位 置 ，lengthO 用 于 判断 文件 的 最 大 尺寸 。 另 外 ， 其 构造 器 还 需要 
第 二 个 参数 (和 C 中 的 fopen0 相 同 ) 用 来 指示 我 们 只 是 “随机 读 ”(r) 还 是 “ 既 读 又 写 ”(rw) 。 
它 并 不 支持 只 写 文件 ， 这 表明 RandomAccessFile 若 是 从 DataInputStream 继 承 而 来 也 可 能 会 运行 
得 很 好 。 

只 有 RandonAccessFile 支 持 搜寻 方法 ， 并 且 只 适用 于 文件 。BufferedInputStream 却 能 允许 
标注 (mark0) 位 置 (其 值 存储 于 内 部 某 个 简单 变量 内 ) 和 重新 设 定位 置 (reset0)， 但 这 些 功 
能 很 有 限 ， 不 是 非常 有 用 。 

在 JDK 1.4 中 ，RandomAccessFile 的 大 多 数 功能 (但 不 是 全 部 ) 由 mio 存 储 映射 文件 所 取代 ， 
本 章 稍 后 会 讲述 。 


18.6 VO 流 的 典型 使 用 方式 


尽管 可 以 通过 不 同 的 方式 组 合 IO 流 类 ， 但 我 们 可 能 也 就 只 用 到 其 中 的 几 种 组 合 。 下 面 的 例 
子 可 以 作为 典型 的 WO 用 法 的 基本 参考 。 在 这 些 示例 中 ， 异 常 处 理 都 被 简化 为 将 异常 传递 给 控制 
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台 ， 但 是 这 只 有 在 小 型 示例 和 工具 中 才 适 用 。 在 代码 中 ， 你 需要 考虑 更 加 复杂 的 错误 处 理 方式 。 
18.6.1 缓冲 输入 文件 

如 果 想 要 打开 一 个 文件 用 于 字符 输入 ， 可 以 使 用 以 String 或 File 对 象 作为 文件 名 的 
FilemputReader。 为 了 提高 速度 ， 我 们 希望 对 那个 文件 进行 缓冲 ， 那 么 我 们 将 所 产生 的 引用 传 
给 一 个 BufferedReader 构 造 器 。 由 于 BufferedReader 也 提供 readLine0 方 法 ， 所 以 这 是 我 们 的 最 
终 对 象 和 进行 读 取 的 接口 。 当 readLine0 将 返回 nul 时， 你 就 达到 了 文件 的 末尾 。 

//: jo/BufferedInputFile. java 

import java.io.*; 


public class BufferedInputFile { 
// Throw exceptions to console: 
public static String 
read(String filename) throws IOException { 
/1 Reading input by lines: 
BufferedReader in = new BufferedReader ( 
new FileReader(filename)) ; 
string s; 
StringBuilder sb = new StringBuilder (); 
while((s = in.readLine())!= null) 
sb.append(s + "\n"); : 
in.close(); 
return sb.toString(); 
} 
public static void main(String(] args) 
throws IOException { 
System. out. print (read("BufferedInputFile.java")); 


$ 
} /* (Execute tó See output) *///:~ 


字符 串 sb 用 来 累积 文件 的 全 部 内 容 (包括 必须 添加 的 换行 符 , 因为 readLine0 已 将 它们 删 掉 ) 。 
最 后 ， 调 用 elose0 关 闭 文件 。 

练习 7， (2) 打开 一 个 文本 文件 ， 每 次 读 取 一 行内 容 。 将 每 行 作为 一 个 String 读 人 ， 并 将 那个 
String 对 象 置 入 一 个 LinkedList 中 。 按 相反 的 顺序 打印 出 LinkedList 中 的 所 有 行 。 

练习 8，(1) 修改 练习 7， 使 要 读 取 的 文件 的 名 字 以 命令 行 参数 的 形式 来 提供 。 

练习 9: (1) 修改 练习 8， 强 制 ArrayList 中 的 所 有 行 都 变 成 大 写 形式 ， 并 将 结果 发 给 
System.out。 

练习 10，(2) 修改 练习 8， 令 它 接受 附加 的 命令 行 参数 ， 用 来 表示 要 在 文件 中 查找 的 单词 。 
打印 出 包含 了 欲 查找 单词 的 所 有 文本 行 。 

练习 11，(2) 在 innerclasses/GreenhouseControllerjava 示 例 中 ，GreenhouseController 包 含 
一 个 硬 编码 的 事件 集 。 修 改 该 程序 ， 使 其 从 一 个 文本 文件 中 读 取 事 件 和 与 它们 相关 联 的 次 数 
[ (不 同 的 难度 级 别 8) : 使 用 工厂 方法 设计 模式 来 构建 事件 一 请 查看 在 www.MindView.net 上 的 
(Thinking in Patterns(with Java) ] 


18.6.2 从 内 存 输入 
在 下 面 的 示例 中 ， 从 BufferedInputFile. read0 读 入 的 String 结 果 被 用 来 创建 一 个 String- 
Reader。 然 后 调用 read0 每 次 读 取 一 个 字符 ， 并 把 它 发 送 到 控制 台 。 


O 在 最 初 的 设计 中 ，close0 被 设 为 在 finalize0 运 行 时 被 调用 ,你 可 以 看 到 finalize0 为 /O 类 定义 了 这 种 方式 。 但 是 ， 
正如 本 书 其 他 地 方 所 讨论 的 那样 ，finalize0 特 性 并 未 像 Java 设 计 者 最 初 设想 的 那样 得 以 实现 《 即 ， 它 的 问题 是 
不 可 恢复 的 )， 因 此 唯一 安全 的 方式 就 是 对 文件 显 式 地 调用 close0。 
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//: io/MemoryInput. java 
import java.io.*; 











public class MemoryInput { 928) 
public static void main(String[] args) 
throws IOException { 
StringReader in = new StringReader( 
BufferedInputFile.read("MemoryInput. java")); 
int c; 
while((¢ = in.read()) != -1) 
System. out. print ((char)c); 








} I+ (Execute to see output) *///:~ 
注意 read0 是 以 int 形 式 返 回 下 一 字 节 ， 因 此 必须 类 型 转换 为 char 才 能 正确 打印 。 
18.6.3 格式 化 的 内 存 输入 

要 读 取 格式 化 数据 ， 可 以 使 用 DataInputStream， 它 是 一 个 面向 字 节 的 MO 类 (不 是 面向 字 
符 的 )。 因 此 我 们 必须 使 用 InputStream 类 而 不 是 Reader 类 。 当 然 ， 我 们 可 以 用 InputStream 以 字 
节 的 形式 读 取 任 何 数据 〈 例 如 一 个 文件 )， 不 过 ， 在 这 里 使 用 的 是 字符 串 。 


//: fo/FormattedMemoryInput.java 
import java.io.*; 


public class FormattedMemoryInput { 
public static void main(String{] args) 
throws IOException { 
try { 
DataInputStream in = new DataInputStream( 
new ByteArrayInputStream( 
BufferedInputF ile. read( 
"FormattedMemoryInput. java") .getBytes())): 
while(true) 
System. out. print ((char) in. readByte()); 
} catch(EOFException e) { 
System.err.printin("End of stream"); 
$ 


} 
} /* (Execute to see output) *///:~ 


必须 为 ByteArrayInputStream 提 供 字 节 数组 ， 为 了 产生 该 数组 String 包 含 了 一 个 可 以 实现 此 
项 工作 的 getBytes() 方 法 。 所 产生 的 ByteArrayInputStrem 是 一 个 适合 传递 给 DataInputStream 的 
InputStream, 

如 果 我 们 从 DataInputStream 用 readByte0 一 次 一 个 字 节 地 读 取 字 符 ， 那 么 任何 字 节 的 值 都 
是 合法 的 结果 ， 因 此 返回 值 不 能 用 来 检测 输入 是 否 结束 。 相 反 ， 我 们 可 以 使 用 available0 方 法 查 
看 还 有 多 少 可 供 存 取 的 字符 。 下 面 这 个 例子 演示 了 怎样 一 次 一 个 字 节 地 读 取 文件 : 


/1: io/TestEOF.java 
// Testing for end of file while reading a byte at a time. 
import java.io.*; 





public class TestEOF { 
public static void main(String[] args) 
throws IOException { 
DataInputStream in = new DataInputStream( 
new BufferedInputStream( 
new FileInputStream(*TestEOF. java"))); 
while(in.available() != 6) 
System. out. print ((char)in.readByte()); 


$ 
/* (Execute to see output) *///:~ 











1930) 








542 FISH 





注意 ，available0 的 工作 方式 会 随 着 所 读 取 的 媒介 类 型 的 不 同 而 有 所 不 同 ， 字 面 意思 就 是 
“在 没有 阻塞 的 情况 下 所 能 读 取 的 字 节 数 "。 对 于 文件 ， 这 意味 着 整个 文件 ， 但 是 对 于 不 同类 型 
的 流 ， 可 能 就 不 是 这 样 的 ， 因 此 要 谨慎 使 用 。 

我 们 也 可 以 通过 捕获 异常 来 检测 输入 的 末尾 。 但 是 ， 使 用 异常 进行 流 控制 ， 被 认为 是 对 异 
常 特性 的 错误 使 用 。 

18.6.4 基本 的 文件 输出 

FileWriter 对 象 可 以 向 文件 写 人 数据 。 首 先 ， 创 建 一 个 与 指定 文件 连接 的 FileWriter。 实 际 
上 ， 我 们 通常 会 用 BufferedWriter 将 其 包装 起 来 用 以 缓冲 输出 (尝试 移 除 此 包装 来 感受 对 性 能 的 
影响 -一 缓冲 往往 能 显著 地 增加 MO 操作 的 性 能 ) 。 在 本 例 中 ， 为 了 提供 格式 化 机 制 ， 它 被 装饰 成 
了 PrintWriter。 按 照 这 种 方式 创建 的 数据 文件 可 作为 普通 文本 文件 读 取 。 


//; 10/BasicFileOutput.java 
‘import java.io.*; 


public class BasicFileOutput { 
static String file = "BasicFileOutput. out"; 
public static void main(String{] args) 
throws IOException { 
BufferedReader in = new BufferedReader ( 
new StringReader( 
Buf feredInputFile.read("BasicFileOutput. java"))); 
PrintWriter out = new Printhriter( 
new BufferedWriter(new FileWriter(file))): 
int lineCount = 1; 
String s; 
while((s = in.readLine()) != null ) 
out.printin(LineCount++ + ": " + s); 
out.close(); 
// Show the stored file: 
System.out .printLn(BufferedinputFile.read(file)) ; 


} /* (Execute to see output) *///:~ 


当 文 本 行 被 写 人 文件 时 ， 行 号 就 会 增加 。 注 意 并 未 用 到 LineNumberInputStream， 因 为 这 
个 类 没有 多 大 帮助 ， 所 以 我 们 没 必要 用 它 。 从 本 例 中 可 以 看 出 ， 记 录 自 己 的 行 号 很 容易 。 

一 旦 读 完 输入 数据 流 ，readLine0 会 返回 null。 我 们 可 以 看 到 要 为 out 显 式 调 用 close0。 如 果 
我 们 不 为 所 有 的 输出 文件 调用 close0 ， 就 会 发 现 缓冲 区 内 容 不 会 被 刷新 清空 ， 那 么 它们 也 就 不 
完整 。 

文本 文件 输出 的 快捷 方式 

Java SE5 在 PrintWriter 中 添加 了 一 个 辅助 构造 器 ， 使 得 你 不 必 在 每 次 希望 创建 文本 文件 并 
向 其 中 写 人 时 ， 都 去 执行 所 有 的 装饰 工作 。 下 面 是 用 这 种 快捷 方式 重 写 的 BasicFileOutputjava; 


//: 10/FileOutputShortcut. java 
import java.io.*; 


public class FileQutputShortcut { 
static String file = “FileQutputShortcut.out”; 
public static void main(Stringl] args) 
throws IOException { 
BufferedReader in = new BufferedReader ( 
new StringReader( 
Buf feredInputFile.read("FileOutputShortcut.java"))); 
// Here's the shortcut: 
PrintWriter out = new PrintWriter (file); 
int LineCount = 1; 
String s; 
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while((s = in.readLine()) != null ) 
out.printin(lineCount++ + *: " + s); 

out.close(); 

// Show the stored file: 

System. out.printtn(BufferedInputFile.read(file)); 


} k (Execute to see output) *///:~ 

你 仍旧 是 在 进行 缓存 ， 只 是 不 必 自 己 去 实现 。 遗 憾 的 是 ， 其 他 常见 的 写 信 任务 都 没有 快捷 
方式 ， 因 此 典型 的 MO 仍旧 包含 大 量 的 宛 余 文本 。 但 是 ， 本 书 所 使 用 的 在 本 章 稍 后 进行 定义 的 
TextFile 工 具 简化 了 这 些 常见 任务 。 

练习 12: (3) 修改 练习 8， 同 样 也 打开 一 个 文本 文件 ， 以 便 将 文本 写 人 其 中 。 将 LinkedList 中 
的 各 行 随同 行 号 一 起 写 人 文件 〈 不 要 试图 使 用 LineNumber 类 ) 。 

练习 13: (3) 修改 BasicFileOutputjava， 以 便 可 以 使 用 LineNumberReader 来 记录 行 数 。 注 
意 继续 使 用 编程 方式 实现 跟踪 会 更 简单 。 

练习 14，(2) 从 BasicFileOutputjava 的 第 四 部 分 开始 ， 编 写 一 个 程序 ， 用 来 比较 有 缓冲 的 和 
无 缓冲 的 IO 方式 在 向 文件 写 入 时 的 性 能 差别 。 


18.6.5 存储 和 恢复 数据 

PrintWriter 可 以 对 数据 进行 格式 化 ， 以 便 人 们 的 阅读 。 但 是 为 了 输出 可 供 另 一 个 “ 流 ” 恢 
复 的 数据 ， 我 们 需要 用 DataOutputStream 写 人 数据 ， 并 用 DataInputStream 恢 复数 据 。 当 然 ， 
这 些 流 可 以 是 任何 形式 ， 但 在 下 面 的 示例 中 使 用 的 是 一 个 文件 ， 并 且 对 于 读 和 写 都 进行 了 缓冲 
处 理 。 注 意 DataOutputStream 和 DataInputStream 是 面向 字 节 的 ， 因 此 要 使 用 InputStream 和 
OutputStream, 


//: 10/StoringAndRecover ingData. java 
import java.io.*; 


public class StoringAndRecoveringData { 
public static void main(Stringl] args) 
throws IOException { 

DataQutputStream out = new Datadutputstream( 

new Buf feredOutputStream( 

new FileQutputStream("Data,txt"))); 

out.writeDouble(3. 14159) ; 
out.writeUTF("That was pi"); 
out wri teDouble (1.41413) ; 
out.writeUTF("Square root of 2"); 
out .close(); 
DataInputStream in = new DataInputStream( 

new BufferedinputStream( 

new FileInputStream("Data. txt"))); 

System.out.printIn(in.readDouble()); 
77 Only readUTF() will recover the 
// Java-UTF String properly: 
System.out.printtn(in. readUTF()): 
System.out .printin(in. readDouble()) ; 
System.out .printin(in.readUTF()); 


} 
} /* Output: 
3.14159 
That was pi 
1.41413 
Square root of 2 
Wm 


如 果 我 们 使 用 DataOutputStream 写 和 数据，Java 保 证 我 们 可 以 使 用 DataInputStream 淮 确 地 
读 取 数据 一 无论 读 和 写 数据 的 平台 多 么 不 同 。 这 一 点 很 有 价值 ， 因 为 我 们 都 知道 ， 人 们 曾经 
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花费 了 大 量 时 间 去 处 理 特定 于 平台 的 数据 问题 。 只 要 两 个 平台 上 都 有 Java， 这 种 问题 就 不 会 再 
RES, 

当 我 们 使 用 DataOutputStream 时 ， 写 字符 串 并 且 让 DataInputStream 能 够 恢复 它 的 唯一 可 
靠 的 做 法 就 是 使 用 UTF-8 编 码 ， 在 这 个 示例 中 是 用 writeUTFO 和 readUTF0 来 实现 的 。UTF-8 是 
一 种 多 字 节 格式 ， 其 编码 长 度 根据 实际 使 用 的 字符 集会 有 所 变化 。 如 果 我 们 使 用 的 只 是 ASCI 或 
者 几乎 都 是 ASCII 字 符 ( 只 占 7 位 )， 那 么 就 显得 极其 浪费 空间 和 带宽 ， 所 以 UTF-8 将 ASCII 字 符 
编码 成 单一 字 节 的 形式 ， 而 非 ASCII 字符 则 编码 成 两 到 三 个 字 节 的 形式 。 另 外 ， 字 符 串 的 长 度 
存储 在 UTF-8 字 符 串 的 前 两 个 字 节 中 。 但 是 ，writeUTFO 和 readUTFO 使 用 的 是 适合 于 Java 的 
UTF-8 变 体 (JDK 文 档 中 有 这 些 方 法 的 详尽 描述 )， 因 此 如 果 我 们 用 一 个 非 Java 程 序 读 取 用 
writeUTFO 所 写 的 字符 串 时 ， 必 须 编写 一 些 特殊 代码 才能 正确 读 取 字符 串 。 

有 了 writeUTFO 和 readUTF0， 我 们 就 可 以 用 DataOutputStream 把 字符 串 和 其 他 数据 类 型 
相 混合 ， 我 们 知道 字符 串 完全 可 以 作为 Unicode 来 存储 ， 并 且 可 以 很 容易 地 使 用 DataInput- 
Stream 来 恢复 它 。 

writeDouble0 将 double 类 型 的 数字 存储 到 流 中 ， 并 用 相应 的 readDouble0 恢 复 它 ( 对 于 其 他 
的 数据 类 型 ， 也 有 类 似 方法 用 于 读 写 )。 但 是 为 了 保证 所 有 的 读 方 法 都 能 够 正常 工作 ， 我 们 必须 
知道 流 中 数据 项 所 在 的 确切 位 置 ， 因 为 极 有 可 能 将 保存 的 double 数 据 作为 一 个 简单 的 字 节 序列 、 
char 或 其 他 类 型 读 入 。 因 此 ， 我 们 必须 要么 为 文件 中 的 数据 采用 固定 的 格式 ， 要 么 将 额外 的 
信息 保存 到 文件 中 ， 以 便 能 够 对 其 进行 解析 以 确定 数据 的 存放 位 置 。 注 意 ， 对 象 序列 化 和 XML 
(本 章 稍 后 都 会 介绍 ) 可 能 是 更 容易 的 存储 和 读 取 复杂 数据 结构 的 方式 。 

练习 15: (4) 在 JDK 文 档 中 查找 DataOutputStream 和 DataInputStream， 以 Storing-And- 
RecoveringData.java 为 基础 ， 创 建 一 个 程序 ， 它 可 以 存储 然后 获取 DataOutputStream 和 
DataInputStream 类 能 够 提供 的 所 有 不 同 的 类 型 。 验 证 它 可 以 准确 地 存储 和 获取 各 个 值 
18.6.6 读 写 随机 访问 文件 

使 用 RandomAeccessFile， 类 似 于 组 合 使 用 了 DataInputStream 和 DataOutputStream (因为 
它 实现 了 相同 的 接口 : Datalinput 和 DataOutput) 。 另 外 我 们 可 以 看 到 ， 利 用 seek0 可 以 在 文件 中 
到 处 移动 ， 并 修改 文件 中 的 某 个 值 。 

在 使 用 RandomAccessFile 时 ， 你 必须 知道 文件 排版 ， 这 样 才能 正确 地 操作 它 。Random- 
AccessFile 拥 有 读 取 基本 类 型 和 UTF-8 字 符 串 的 各 种 具体 方法 。 下 面 是 示例 : 


//: io/UsingRandomaccessFile. java 
import java.io.*; 


public class UsingRandomAccessFite { 
static String file = "rtest.dat"; 
static void display() throws IOException { 
RandomAccessFile rf = new RandomAccessFile(fite, "r"); 


for(int i = 0; i <7; i++) 
‘System. out. printin( 
“Value "+i +": "+ rf.readDouble()); 


System. out.printin(rf.readUTF()); 
rf.close(); 
} 
public static void main(String[] args) 
throws IOException { 
RandomAccessFile rf = new RandomAccessFile(file, "rw"); 


O XML 是 另 一 种 方式 ， 可 以 解决 在 不 同 的 计算 平台 之 间 移 动 数据 ， 而 不 依赖 于 所 有 平台 上 都 有 Java 这 一 问题 。 
XML 将 在 本 章 稍 后 进行 介绍 。 
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for(int i = @; i < 7; i++) 
rf.writeDouble(i*l. 414); 

rf.writeUTF("The end of the file"); 

rf.close(): 

display(); 

rf = new RandomAccessFile(file, "rw"); 

rf.seek(5*8); 

rf.writeDouble (47.0901) ; 

rf.close(); 

display(); 


} 
} /* Output: 
Value 0: 0.0 
Value 1: 1.414 
Value 2: 2.828 
Value 3: 4.242 
Value 4: 5.656 
Value 5: 7.069999999999999 
Value 6: 8.484 
The end of the file 
Value 0: 0.0 
Value 1: 1.414 
Value 2: 2.828 
Value 3: 4.242 
Value 4: 5.656 
Value 5: 47.0001 
Value 6: 8.484 
The end of the file 
Wha 


display() 方 法 打开 了 一 个 文件 ， 并 以 double 值 的 形式 显示 了 其 中 的 七 个 元 素 。 在 main0 中 ， 
首先 创建 了 文件 ， 然 后 打开 并 修改 了 它 。 因 为 double 总 是 8 字 节 长 ， 所 以 为 了 用 seek0 查 找 第 5 个 
双 精 度 值 ， 你 只 需 用 5*8 来 产生 查找 位 置 。 

正如 先前 所 指 ，RandomAccessFile 除 了 实现 DataInput 和 DataOutput 接 口 之 外 ， 有 效 地 与 
UO 继承 层次 结构 的 其 他 部 分 实现 了 分 离 它 不 支持 装饰 ， 所 以 不 能 将 其 与 InputStream 及 
OutputStream 子 类 的 任何 部 分 组 合 起 来 。 我 们 必须 假定 RandomAeccessFile 已 经 被 正确 缓冲 ， 
为 我 们 不 能 为 它 添加 这 样 的 功能 。 

可 以 自行 选择 的 是 第 二 个 构造 器 参数 : 我 们 可 指定 以 “只 读 ”(r) 方式 或 “ 读 写 ”(rw) 方 
式 打开 一 个 RandomAccessFile 文 件 。 

你 可 能 会 考虑 使 用 “内 存 映射 文件 ”来 代替 RandomAccessFile。 

练习 16，(4) 在 JDK 文 档 中 查找 RandomAccessFile， 以 UsingRandomAccessFilejava 为 基础 ， 
创建 一 个 程序 ， 它 可 以 存储 然后 获取 RandomAccessFile 类 能 够 提供 的 所 有 不 同 的 类 型 。 验 证 它 
可 以 准确 地 存储 和 获取 各 个 值 。 
18.6.7 管道 流 

PipedInputStream、PipedOutputStream、PipedReader 及 PipedWriter 在 本 章 只 是 简单 地 提 
到 。 但 这 并 不 表明 它们 没有 什么 用 处 ， 它 们 的 价值 只 有 在 我 们 开始 理解 多 线程 之 后 才 会 显现 ， 
因为 管道 流 用 于 任务 之 间 的 通信 。 这 些 在 第 21 章 会 用 一 个 示例 进行 讲述 。 


18.7 文件 读 写 的 实用 工具 


一 个 很 常见 的 程序 化 任务 就 是 读 取 文件 到 内 存 ， 修 改 ， 然 后 再 写 出 。Java IO 类 库 的 问题 之 
一 就 是 : 它 需要 编写 相当 多 的 代码 去 执行 这 些 常用 操作 一 一 没有 任何 基本 的 帮助 功能 可 以 为 我 们 
做 这 一 切 。 更 糟糕 的 是 ， 装 饰 器 会 使 得 要 记 住 如 何 打开 文件 变 成 一 件 相当 困难 的 事 。 因 此 ， 在 
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我 们 的 类 库 中 添加 帮助 类 就 显得 相当 有 意义 ， 这 样 就 可 以 很 容易 地 为 我 们 完成 这 些 基 本 任务 。 
Java SE5 在 PrintWriter 中 添加 了 方便 的 构造 器 ， 因 此 你 可 以 很 方便 地 打开 一 个 文本 文件 进行 写 人 
操作 。 但 是 ， 还 有 许多 其 他 的 常见 操作 是 你 需要 反复 执行 的 ， 这 就 使 得 消除 与 这 些 任务 相关 联 
的 重复 代码 就 显得 很 有 意义 了 。 

下 面 的 TextFile 类 在 本 书 前 面 的 示例 中 就 已 经 被 用 来 简化 对 文件 的 读 写 操作 了 。 它 包含 的 
static 方 法 可 以 像 简单 字符 串 那 样 读 写 文本 文件 ， 并 且 我 们 可 以 创建 一 个 TextFile 对 象 ， 它 用 一 
个 ArrayList 来 保存 文件 的 若干 行 (如 此 ， 当 我 们 操纵 文件 内 容 时 ， 就 可 以 使 用 ArrayList 的 所 
有 功能 ) 。 


//: net/mindview/util/TextFile.java 

// Static functions for reading and writing text files as 
// a single string, and treating a file as an ArrayList. 
package net.mindview.util; 

import java.io.*; 

import java.util.*: 


public class TextFile extends ArrayList<String> { 
// Read a file as a single string: 
public static String read(String fileName) { 
StringBuilder sb = new StringBuilder(); 
try { 
BufferedReader in= new BufferedReader (new FileReader ( 
new File(fileName) .getAbsoluteFile())): 
try { 
String s: 
while((s = in.readLine()) != null) { 
sb. append(s); 
sb. append("\n") ; 


} 
} finally { 
in.close(): 


y 
入 catch(I0Exception e) { 
throw new RuntimeException(e) ; 


return sb, toString); 


// Write a single file in one method call: 
public static void write(String fileName, String text) { 
try { 
PrintWriter out = new PrintWriter( 
new File(fileName).getAbsoluteFile()); 
try { 
out. print (text); 
} finally { 
out .close(); 


} 
} catch(IOException e) { 

throw new RuntimeException(e); 
} 


// Read a file, split by any regular expression: 
public TextFile(String fileName, String splitter) { 
super (Arrays.asList(read(fileName).split(splitter))); 
// Regular expression split() often leaves an empty 
// String at the first position: 
if (get (®)..equals("")) remove(6); 


} 

// Normally read by lines: 

public TextFile(String fileName) 《 
this (fileName, "\n"): 


} 
public void write(String fileName) { 
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try { 

PrintWriter out = new PrintWriter( 

new File(fileName) .getAbsoluteFile()); 
try { 

for (String item : this) 

out. printin(item); 

} finally { 

out.close(): 


$ 
} catch(IOException e) { 
throw new RuntimeException(e); 
} 
} 
// Simple test: 
public static void main(String{] args) { 
String file = read("TextFile. java"); 
write("test.txt", file); 
TextFile text = new TextFile(*test.txt"); 
text.write(*test2.txt"); 
// Break into unique sorted list of words: 
TreeSet<String> words = new TreeSet<String>( 
new TextFile("TextFile.java". "\\W+")): 
// Display the capitalized words: 
System.out.printin(words.headSet("a")) 


} 
} /* Output: 
[@, ArrayList, Arrays, Break, BufferedReader, 
Bufferedwriter, Clean, Display, File, FileReader, 
FileWriter, IOException, Normally. Output, Printwriter, 
Read, Regular, RuntimeException, Simple, Static, String, 
StringBuilder, System, TextFile, Tools, TreeSet, W, Write] 
Whim. 


read0 将 每 行 添加 到 StringBuffer， 并 且 为 每 行 加 上 换行 符 ， 因 为 在 读 的 过 程 中 换行 符 会 被 
去 除 掉 。 接 着 返回 一 个 包含 整个 文件 的 字符 种。write0 打 开 文 本 并 将 其 写 和 文件。 在 这 两 个 方 
法 完成 时 ， 都 要 记 着 用 close0 关 闭 文件 。 

注意 , 在 任何 打开 文件 的 代码 在 finally 子 句 中 ， 作 为 防卫 措施 都 添加 了 对 文件 的 close0 调 用 ， 
以 保证 文件 将 会 被 正确 关闭 。 

这 个 构造 器 利用 read0 方 法 将 文件 转换 成 字符 串 ， 接 着 使 用 String.split0 以 换行 符 为 界 把 结 
果 划 分 成 行车 要 频繁 使 用 这 个 类 ， 我 们 可 以 重 写 此 构造 器 以 提高 性 能 )。 遗 憾 的 是 没有 相应 
的 连接 (join) 方法 ， 所 以 那个 非 静 态 的 write0 方 法 必须 一 行 一 行 地 输出 这 些 行 。 因 为 这 个 类 
希望 将 读 取 和 写 和 人 文件 的 过 程 简单 化 ， 因 此 所 有 的 IOException 都 被 转型 为 RuntimeException 
因此 用 户 不 必 使 用 try-eateh 语 句 块 。 但 是 ， 你 可 能 需要 创建 另 一 种 版 本 将 IOException 传 递 给 调 
用 者 。 

在 main0 方 法 中 ， 通 过 执行 一 个 基本 测试 来 确保 这 些 方法 正常 工作 。 尽 管 这 个 程序 不 需要 
创建 许多 代码 ， 但 使 用 它 会 节约 大 量 时 间 ， 它 会 使 你 变 得 很 轻松 ， 在 本 章 后 面 一 些 例子 中 就 可 
以 感受 到 这 一 点 。 i 

另 一 种 解决 读 取 文件 问题 的 方法 是 使 用 在 Java SE5 中 引入 的 java.util.Scanner 类 。 但 是 ， 这 
只 能 用 于 读 取 文件 ， 而 不 能 用 于 写 人 文件 ， 并 且 这 个 工具 (你 会 注意 到 它 不 在 java.io 包 中 ) £ 
要 是 设计 用 来 创建 编程 语言 的 扫描 器 或 “小 语言 ”的 。 

练习 17: (4) 用 TextFile 和 Map<Character, Integer> 创 建 一 个 程序 ， 它 可 以 对 在 一 个 文件 中 
所 有 不 同 的 字符 出 现 的 次 数 进行 计数 。( 因 此 如 果 在 文件 中 字母 a 出 现 了 12 次 ， 那 么 在 Map 中 与 
包含 8 的 Character 相 关联 的 Integer 就 包含 12) 。 

练习 18: (1) 修改 TextFilejava， 使 其 可 以 将 IOException 传 递 给 调用 者 。 








938| 














548 RISE 





18.7.1 读 取 二 进 制 文件 
这 个 工具 与 TextFile 类 似 ， 因 为 它 简化 了 读 取 二 进 制 文件 的 过 程 : 


//: net/mindview/util/BinaryFile.java 

// Utility for reading files in binary form. 
package net.mindview.util; 

import java.io.*; 


public class BinaryFile { 
public static byte[] read(File bFile) throws IOException{ 
BufferedInputStream bf = new BufferedInputStream( 
new FileInputStream(bFile)); 
try { 
byte[] data = new byte[bf.available()]; 
bf. read(data) ; 
return data; 
} finally { 
bf.close(); 
} 
} 
public static byte[] 
read(String bFile) throws IOException { 
return read(new File(bFile).getAbsoluteFile()); 


} 

} Mi~ 

其 中 一 个 重 载 方法 接受 File 参 数 ， 第 二 个 重 载 方法 接受 表示 文件 名 的 String 参 数 。 这 两 个 方 
法 都 返回 产生 的 byte 数 组 。available( 方 法 被 用 来 产生 恰当 的 数组 尺寸 ， 并 且 read() 方 法 的 特定 
的 重 载 版 本 填充 了 这 个 数组 。 

练习 19: (2) 用 BinaryFile 和 Map<Byte, Integer> 创 建 一 个 程序 ， 它 可 以 对 在 一 个 文件 中 所 
有 不 同 的 字 节 出 现 的 次 数 进行 计数 。 

练习 20: (4) 用 Directory.walk0 和 BinaryFile 来 验证 在 某 个 目录 树 下 的 所 有 的 .class 文 件 都 是 
以 十 六 进 制 字符 “CAFEBABE” 开 头 的 。 


18.8 标准 MO 


标准 MO 这 个 术语 参考 的 是 Unix 中 “程序 所 使 用 的 单一 信息 流 ” 这 个 概念 〈 在 Windows 和 其 
他 许多 操作 系统 中 ， 也 有 相似 形式 的 实现 )。 程 序 的 所 有 输入 都 可 以 来 自 于 标准 输入 ， 它 的 所 有 
输出 也 都 可 以 发 送 到 标准 输出 ， 以 及 所 有 的 错误 信息 都 可 以 发 送 到 标准 错误 。 标 准 1/O 的 意义 在 
于 : 我 们 可 以 很 容易 地 把 程序 串联 起 来 ， 一 个 程序 的 标准 输出 可 以 成 为 另 一 程序 的 标准 输入 。 
这 真是 一 个 强大 的 工具 。 
18.8.1 从 标准 输入 中 读 取 

按照 标准 MO 模型 ，Java 提 供 了 System.in、System.out 和 System.err。 在 整 本 书 里 ， 我 们 已 经 
看 到 了 怎样 用 System.out 将 数据 写 出 到 标准 输出 ， 其 中 System.out 已 经 事先 被 包装 成 了 
printStream 对 象 。System.err 同 样 也 是 PrintStream ， 但 System.in 却 是 一 个 没有 被 包装 过 的 未 经 
加 工 的 InputStream。 这 意味 尽管 我 们 可 以 立即 使 用 System.out 和 System.err， 但 是 在 读 取 
System.in 之 前 必须 对 其 进行 包装 。 

通常 我 们 会 用 readLine0 一 次 一 行 地 读 取 输 入 ， 为 此 ， 我 们 将 System.in 包 装 成 Buffered- 
Reader 来 使 用 这 要 求 我 们 必须 用 InputStreamReader 把 System.in 转 换 成 Reader。 下 面 这 个 例子 
将 直接 回 显 你 所 输入 的 每 一 行 。 


1/: io/Echo.java 
// How to read from standard input. 
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77 {RunByHand} 
‘import java.io.*: 


public class Echo { 
public static void main(String(] args) 
throws IOException { 
BufferedReader stdin = new BufferedReader ( 
new InputStreamReader (System. in)); 








String s; 941 








while((s = stdin.readLine()) != null && s.length()!= @) 
System.out.println(s); 
// An empty Line or Ctrl-Z terminates the program 


} 

yh 

使 用 异常 规范 是 因为 readLine0 会 抛 出 IJOException。 注 意 ，System.in 和 大 多 数 流 一 样 , 通 
常 应 该 对 它 进行 缓冲 。 

练习 21: (1) 写 一 个 程序 ， 它 接受 标准 输入 并 将 所 有 字符 转换 为 大 写 ， 然 后 将 结果 写 信 到 
标准 输出 流 中 。 将 文件 的 内 容重 定向 到 该 程序 中 ( 重 定向 的 过 程 会 根据 操作 系统 的 不 同 而 有 所 
变化 )。 
18.8.2 将 System.out 转 换 成 PrintWriter 

System.out 是 一 个 PrintStream ， 而 PrintStream 是 一 个 OutputStream。PrintWriter 有 一 个 
可 以 接受 OutputStream 作 为 参数 的 构造 器 。 ， 只 要 需要 ， 就 可 以 使 用 那个 构造 器 把 
System.out 转换 成 PrintWriter: 





/1/: 40/ChangeSystemOut. java 
// Turn System.out into a PrintWriter. 
import java.io.*; 
public class ChangeSystemOut { | 

public static void main(String[] args) { 


PrintWriter out = new Printhriter(System.out, true); 
out.printin("Hello, world"); 


二 
} /* Output: 
Hello, world 
Whim 


重要 的 是 要 使 用 有 两 个 参数 的 PrintWriter 的 构造 器 ， 并 将 第 二 个 参数 设 为 tue， 以 便 开 启 自 
动 清空 功能 ， 否 则 ， 你 可 能 看 不 到 输出 。 
18.8.3 标准 UVO 重 定向 

Java 的 System 类 提供 了 一 些 简单 的 静态 方法 调用 ， 以 允许 我 们 对 标准 输入 、 输 出 和 错误 IO 
流 进行 重 定向 : 

setIn(InputStream) 

setOut(PrintStream) 

setErr(PrintStream) 

如 果 我 们 突然 开始 在 显示 器 上 创建 大 量 输 出 ， 而 这 些 输出 滚动 得 太 快 以 至 于 无 法 阅读 时 ， 
重 定向 输出 就 显得 极为 有 用 。。 对 干 我 们 想 重复 测试 某 个 特定 用 户 的 输入 序列 的 命令 行程 序 来 说 ， 
重 定向 输入 就 很 有 价值 。 下 例 简单 演示 了 这 些 方法 的 使 用 ; 


//: i0/Redirecting. java 
// Demonstrates standard 1/0 redirection. 


$ 
Š 


O 第 22 章 展示 了 一 种 更 方便 的 解决 方案 : 一 个 GUI 程序 ,具有 带 滚动 的 文本 区 域 。 











550 #18 È 








943 











[944| 








import java.io.*; 


public class Redirecting { 
public static void main(String[] args) 
throws IOException { 
PrintStream console = System. out; 
BufferedInputStream in = new BufferedInputStream( 
new FileInputStream("Redirecting.java")): 
PrintStream out = new PrintStream( 
new BufferedOutputStream( 
new FileQutputStream("test.out"))):; 
System. setIn(in) ; 
System. setOut (out) ; 
System. setErr (out); 
BufferedReader br = new BufferedReader ( 
new InputStreamReader (System. in)) ; 
String s; 
while((s = br.readLine()) != null) 
System.out.printin(s); 
out.close(); // Remember this! 
System. setOut (console); 


) 
VM 
这 个 程序 将 标准 输入 附 接 到 文件 上 ， 并 将 标准 输出 和 标准 错误 重 定向 到 另 一 个 文件 。 注 意 ， 
它 在 程序 开头 处 存储 了 对 最 初 的 System.out 对 象 的 引用 ， 并 且 在 结尾 处 将 系统 输出 恢复 到 了 该 对 
RE. 
J/O 重 定向 操纵 的 是 字 节 流 ， 而 不 是 字符 流 ， 因 此 我 们 使 用 的 是 InputStream 和 Output- 
Stream， 而 不 是 Reader 和 Writer。 


18.9 进程 控制 


你 经 常会 需要 在 Java 内 部 执行 其 他 操作 系统 的 程序 ， 并 且 要 控制 这 些 程序 的 输入 和 输出 。 
Java 类 库 提供 了 执行 这 些 操作 的 类 。 

一 项 常见 的 任务 是 运行 程序 ， 并 将 产生 的 输出 发 送 到 控制 人 台 。 本 节 包 含 了 一 个 可 以 简化 这 
项 任务 的 实用 工具 。 在 使 用 这 个 实用 工具 时 ， 可 能 会 产生 两 种 类 型 的 错误 ， 普 通 的 导致 异常 的 
错误 一 一 对 这 些 错误 我 们 只 需 重新 抛 出 一 个 运行 时 异常 ， 以 及 从 进程 自身 的 执行 过 程 中 产生 的 
错误 ， 我 们 希望 用 单独 的 异常 来 报告 这 些 错误 : 

//: net/mindview/util/OSExecuteException. java 

package net .mindview.util: 


public class OSExecuteException extends RuntimeException { 
public OSExecuteException(String why) { super(why); } 
} Mi~ 


要 想 运行 一 个 程序 ， 你 需要 向 OSExecute.command0 传 递 一 个 command 字 符 串 ， 它 与 你 在 
控制 台 上 运行 该 程序 所 键入 的 命令 相同 。 这 个 命令 被 传递 给 java.lang.ProcessBuilder 构 造 器 ( 它 
要 求 这 个 命令 作为 一 个 String 对 象 序列 而 被 传递 )， 然 后 所 产生 的 ProcessBuilder 对 象 被 启动 : 


//: net/mindview/util/0SExecute.java 
// Run an operating system command 

// and send the output to the console. 
package net.mindview.util; 

import java.io.*; 





public class OSExecute { 
public static void command(String command) { 
boolean err = false; 
try { 
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Process process = 
new ProcessBuilder (command.split(" *)).start(; 
BufferedReader results = new BufferedReader ( 
new InputStreamReader (process. getInputStream())); 
String s: 
while((s = results.readLine())!= null) 
System.out.printtn(s) ; 
BufferedReader errors = new BufferedReader ( 
new InputStreamReader (process. getErrorStream())); 
// Report errors and return nonzero value 
// to calling process if there are problems: 
while((s = errors.readLine())!= null) { 
System.err.printin(s); 
err = true; 


$ 
catch(Exception e) { 
// Compensate for Windows 2000, which throws an 
// exception for the default command line: 
if (command. startsWith("CMD /C")) 
command(:'CMD /C ”+ command); 
else 
throw new RuntimeException(e) ; 


Vetere) 
throw new OSExecuteException("Errors executing ”+ 
command) ; 
) 

} Ii~ 

为 了 捕获 程序 执行 时 产生 的 标准 输出 流 ， 你 需要 调用 getInputStream()， 这 是 因为 
InputStream 是 我 们 可 以 从 中 读 取信 息 的 流 。 从 程序 中 产生 的 结果 每 次 输出 一 行 ， 因 此 要 使 用 
readLine0 来 读 取 。 这 里 这 些 行 只 是 直接 被 打印 了 出 来 ， 但 是 你 还 可 能 希望 从 command0 中 捕获 
和 返回 它们 。 该 程序 的 错误 被 发 送 到 了 标准 错误 流 ， 并 且 通 过 调用 getErrotStream() 得 以 捕获 。 
如 果 存在 任何 错误 ， 它 们 都 会 被 打印 并 且 会 抛 出 OSExecuteException， 因 此 调用 程序 需要 处 理 
这 个 问题 。 

下 面 是 展示 如 何 使 用 OSExecute 的 示例 : 


//: io/0SExecuteDemo .java 
// Demonstrates standard I/0 redirection. 
import net.mindview.util.*; 





public class OSExecuteDemo { 
public static void main(String{] args) { 
OSExecute.command("javap OSExecuteDemo") ; 


} 
} /* Output: 
Compiled from "OSExecuteDemo. java” 
public class OSExecuteDemo extends java.lang.Object{ 
public OSExecuteDemo() ; 
public static void main(java.lang.String(]); 


} 

H~ 
这 里 使 用 了 javap 反 编译 器 ( 随 JDK 发 布 ) 来 反 编译 该 程序 。 

练习 22: (5) 修改 0SExecutejava， 使 其 不 打印 标准 输出 流 ， 而 是 以 List 或 多 个 String 的 方法 
返回 执行 程序 后 的 结果 。 演 示 对 这 个 实用 工具 的 新 版 本 的 使 用 方式 。 


18.10 新 MO 


IDK 1.4 的 java.nio.* 包 中 引入 了 新 的 Javal/O 类 库 ， 其 目的 在 于 提高 速度 。 实 际 上 ， 旧 的 WO 
包 已 经 使 用 nio 重 新 实现 过 ， 以 便 充分 利用 这 种 速度 提高 ， 因 此 ， 即 使 我 们 不 显 式 地 用 nio 编 写 代 
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码 ， 也 能 从 中 受益 。 速 度 的 提高 在 文件 0 和 网 络 UO 中 都 有 可 能 发 生 ， 我 们 在 这 里 只 研究 前 者 
对 于 后 者 ， 在 《Thinking in Enterprise Java》 中 有 论述 。 

速度 的 提高 来 自 于 所 使 用 的 结构 更 接近 于 操作 系统 执行 WO 的 方式 : 通道 和 缓冲 器 。 我 们 可 
以 把 它 想像 成 一 个 煤矿 ， 通 道 是 一 个 包含 煤层 (数据 ) 的 矿藏 ， 而 缓冲 器 则 是 派送 到 矿藏 的 卡 
车 。 卡 车 载 满 煤炭 而 归 ， 我 们 再 从 卡车 上 获得 煤炭 。 也 就 是 说 ， 我 们 并 没有 直接 和 通道 交互 
我 们 只 是 和 缓冲 器 交互 ， 并 把 缓冲 器 派送 到 通道 。 通 道 要 么 从 缓冲 器 获得 数据 ， 要 么 向 缓冲 器 
发 送 数据 。 

唯一 直接 与 通道 交互 的 缓冲 器 是 ByteBuffer 一 一 也 就 是 说 ， 可 以 存储 未 加 工 字 节 的 缓冲 器 。 
当 我 们 查询 JDK 文 档 中 的 java.nio.ByteBuffer 时 ， 会 发 现 它 是 相当 基础 的 类 : 通过 告知 分 配 多 少 
存储 空间 来 创建 一 个 ByteBuffer 对 象 ， 并 且 还 有 一 个 方法 选择 集 ， 用 于 以 原始 的 字 节 形式 或 基本 
数据 类 型 输出 和 读 取 数据 。 但 是 ， 没 办 法 输出 或 读 取 对 象 ， 即 使 是 字符 串 对 象 也 不 行 。 这 种 处 
理 虽 然 很 低级 ， 但 却 正 好 ， 因 为 这 是 大 多 数 操作 系统 中 更 有 效 的 映射 方式 。. 

人 旧 IO 类 库 中 有 三 个 类 被 修改 了 ， 用 以 产生 FileChannel。 这 三 个 被 修改 的 类 是 
FileInputStream、FileOutputStream 以 及 用 于 既 读 又 写 的 RandomAccessFile。 注 意 这 些 是 字 节 操 
纵 流 ， 与 低层 的 nio 性 质 一 致 。Reader 和 Writer 这 种 字符 模式 类 不 能 用 于 产生 通道 ， 但 是 
java.nio.channels.Channels 类 提供 了 实用 方法 ， 用 以 在 通道 中 产生 Reader 和 Writer。 

下 面 的 简单 实例 演示 了 上 面 三 种 类 型 的 流 ， 用 以 产生 可 写 的 、 可 读 可 写 的 及 可 读 的 通道 。 


//: io/GetChannel . java 

// Getting channels from streams 
import java.nio.*; 

import java.nio.channels.*; 
import java.io.*; 


public class GetChannet { 
private static final int BSIZE = 1024; 
public static void main(String(] args) throws Exception { 
// Write a file: 
FileChannel fc = 
new FileOutputStream("data.txt").getChannel(); 
fe.write(ByteBuffer.wrap("Some text ".getBytes())); 
fc.close(); 
// Add to the end of the file: 
fc = 
new RandomAccessFile("data.txt", “rw").getChannel(); 
fc.position(fc.size()); // Move to the end 
fc. write (ByteBuffer.wrap("Some more”. getBytes())); 
fe.close(); 
// Read the file: 
fc = new FileInputStream(“data. txt”) .getChannel(); 
ByteBuffer buff = ByteBuffer.allocate(BSIZE) ; 
fc. read (buff) ; 
buff. flipO: 
while (buff .hasRemaining()) 
System.out.print((char)buff.get()); 
} 
} /* Output: 
Some text Some more 
#171:~ 


对 于 这 里 所 展示 的 任何 流 类 ，getChannel0 将 会 产生 一 个 FileChannel。 通 道 是 一 种 相当 基 
础 的 东西 ， 可 以 向 它 传送 用 于 读 写 的 ByteBuffer， 并 且 可 以 锁定 文件 的 某 些 区 域 用 于 独占 式 访问 


© 此 部 分 内 容 由 Chintan Thakker 提 供 。 
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( 稍 后 讲述 )。 

将 字 节 存放 于 ByteBuffer 的 方法 之 一 是 : 使 用 一 种 “put” 方 法 直接 对 它们 进行 填充 ， 填 人 
一 个 或 多 个 字 节 ， 或 基本 数据 类 型 的 值 。 不 过 ， 正 如 所 见 ， 也 可 以 使 用 warp0 方 法 将 已 存在 的 
字 节 数组 “包装 ”到 ByteBuffer 中 。 一 旦 如 此 ， 就 不 再 复制 底层 的 数组 ， 而 是 把 它 作 为 所 产生 的 
ByteBuffer 的 存储 器 ， 我 们 称 之 为 数组 支持 的 ByteBuffer。 

data.txt 文 件 用 RandomAccessFile 被 再 次 打开 。 注 意 我 们 可 以 在 文件 内 随处 移动 FileChannel; 
在 这 里 ， 我 们 把 它 移 到 最 后 ， 以 便 附 加 其 他 的 写 操作 。 

对 于 只 读 访问 ， 我 们 必须 显 式 地 使 用 静态 的 allocate0 方 法 来 分 配 ByteBuffer。nio 的 目标 就 
是 快速 移动 大 量 数据 ， 因 此 ByteBuffer 的 大 小 就 显得 尤为 重要 一 一 实际 上 ， 这 里 使 用 的 IK 可 能 比 
我 们 通常 要 使 用 的 小 一 点 (必须 通过 实际 运行 应 用 程序 来 找到 最 佳 尺寸) 。 

甚至 达到 更 高 的 速度 也 有 可 能 ， 方 法 就 是 使 用 allocateDirect0 而 不 是 allocate0 ， 以 产生 一 
个 与 操作 系统 有 更 高 焕 合 性 的 “直接 ”缓冲 器 。 但 是 ， 这 种 分 配 的 开支 会 更 大 ， 并 且 具 体 实现 
也 随 操作 系统 的 不 同 而 不 同 ， 因 此 必须 再 次 实际 运行 应 用 程序 来 查看 直接 缓冲 是 否 可 以 使 我 们 
获得 速度 上 的 优势 。 

一 旦 调用 read0 来 告知 FileChannel 向 ByteBuffer 存 储 字 节 ， 就 必须 调用 缓冲 器 上 的 flip0， 让 
它 做 好 让 别人 读 取 字 节 的 准备 (是 的 ， 这 似乎 有 一 点 拙劣 ， 但 是 请 记 住 ， 它 是 很 拙劣 的 ， 但 却 
适用 于 获取 最 大 速度 )。 如 果 我 们 打算 使 用 缓冲 器 执行 进一步 的 read0 操 作 ， 我 们 也 必须 得 调用 
clear0 来 为 每 个 read0 做 好 准备 。 这 在 下 面 这 个 简单 文件 复制 程序 中 可 以 看 到 ， 


/1/; fo/ChannetCopy.java 
7/ Copying a file using channels and buffers 
// (Args: ChannelCopy.java test.txt} 

import java.nio.*; 

import java.nio.channels.*; 

import java.io.*; 





public class ChannelCopy { 
private static final int BSIZE = 1024; 
public static void main(String{] args) throws Exception { 
if(args.length != 2) { 
System.out.printin("arguments: sourcefile destfile"); 
System.exit(1); 


} 
FileChannel 
in = new FileInputStream(args([@]).getChannel(), 
out = new FileOutputStream(args[1]) .getChannel(); 
ByteBuffer buffer = ByteBuffer.allocate(BSIZE) ; 
while(in.read(buffer) != -1) { 
buffer.flip(); // Prepare for writing 
out.write(buffer) 
buffer.clear(); // Prepare for reading 
} 


} 
} ii~ 


, 可 以 看 到 ， 打 开 一 个 FileChannel 以 用 于 读 ， 而 打开 另 一 个 以 用 于 写 。ByteBuffer 被 分 配 了 
空间 ， 当 FileChannel.read0 返 回 -1 时 (一 个 分 界 符 ， 考 庸 置疑， 它 源 于 Unix 和 C) ， 表 示 我 们 已 
经 到 达 了 输入 的 末尾 。 每 次 read0 操 作 之 后 ， 就 会 将 数据 输入 到 缓冲 器 中 ，flip0 则 是 准备 缓冲 器 
以 便 它 的 信息 可 以 由 write0 提 取 。write0 操 作 之 后 ， 信 息 仍 在 缓冲 器 中 ， 接 着 clear0 操 作 则 对 所 
有 的 内 部 指针 重新 安排 ， 以 便 缓冲 器 在 另 一 个 read0 操 作 期 间 能 够 做 好 接受 数据 的 准备 。 

然而 ， 上 面 那个 程序 并 不 是 处 理 此 类 操作 的 理想 方式 。 特 殊 方法 transferTo0 和 transferFrom0) 
则 允许 我 们 将 一 个 通道 和 另 一 个 通道 直接 相连 : 
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//: io/TransferTo. java 
// Using transferTo() between channels 
// {Args: TransferTo. java TransferTo. txt} 


hannels.*; 





import java.ni 
import java.io 





public class TransferTo { 
public static void main(String{] args) throws Exception { 
if (args. length ) { 
System.out.printin("arguments: sourcefile destfile"); 
System.exit(1); 








$ 
FiteChannet 
in = new FileInputStream(args(6]).getChannel(), 
out = new FileQutputStream(args(1]).getChannel(); 
in.transferTo(®, in.size(), out); 
11 Or: 
1/ out.transterFrom(in, @, in.size()): 
} 
Yh 
虽然 我 们 并 不 是 经 常 做 这 类 事情 ， 但 是 了 解 这 一 点 还 是 有 好 处 的 。 


18.10.1 转换 数据 

回 过 头 看 GetChannel.java 这 个 程序 就 会 发 现 ， 为 了 输出 文件 中 的 信息 ， 我 们 必须 每 次 只 读 
取 一 个 字 节 的 数据 ， 然 后 将 每 个 byte 类 型 强制 转换 成 char 类 型 。 这 种 方法 似乎 有 点 原始 一 如 果 
我 们 查看 一 下 java.nio.CharBuffer 这 个 类 ， 将 会 发 现 它 有 一 个 toString0 方 法 是 这 样 定义 的 :“ 返 
回 一 个 包含 缓冲 器 中 所 有 字符 的 字符 串 。” 既 然 ByteBuffer 可 以 看 作 是 具有 asCharBuffer( 方 法 
的 CharBuffer， 那 么 为 什么 不 用 它 呢 ? 正如 下 面 的 输出 语句 中 第 一 行 所 见 ， 这 种 方法 并 不 能 解 
决 问题 : 


//: io/BufferToText.java 
// Converting text to and from ByteBuffers 
import java.nio.*; 
import java.nio. channel 
import java.nio.charset 
import java. io.*; 








public class BufferToText { 
private static final int BSIZE = 1024; 
public static void main(String[] args) throws Exception { 
FileChannel fc = 
new FileOutputStream("data2.txt").getChannel(): 
fc.write(ByteBuffer.wrap("Some text”. getBytes())); 
fe. close(); 
fc = new FileInputStream("data2.txt").getChannel(); 
ByteBuffer buff = ByteBuffer.allocate(BSIZE) ; 
fc. read (buff); 
buff. flip(); 
71 Doesn't work: 
System. out. printin(buff.asCharBuffer()); 
17 Decode using this system's default Charset: 
buff .rewind() ; 
String encoding = System.getProperty("file.encoding") ; 
System. out.printin("Decoded using " + encoding + *: " 
+ Charset. forName (encoding) .decode(buff)) ; 
// Or, we could encode with something that will print: 
fc = new FileOutputStream("data2. txt") .getChannel (); 
fc. write (ByteBuffer .wrap 
"Some text". getBytes (“UTF-16BE"))); 
fc.close(); 
// Now try reading again: 
fc = new FileInputStream("data2.txt").getChannel(); 
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buff.clear(); 
fe.read(buff) : 
buff.flip() 
System.out.println(buff.asCharBuffer ()); 
// Use a CharBuffer to write through: 
fc = new FileOutputStream("data2. txt") .getChannet() ; 
buff = ByteBuffer.allocate(24); // More than needed 
buff.asCharBuffer(),put("Some text"); 
fe.write (buff) ; 
fe.close(); 
// Read and display: 
fc = new FileInputStream("data2. txt”) .getChannel(); 
buff.clear(): 
fc. read(buff) ; 
buff. flip; 
System. out.printin(buff.asCharBuffer()); 
J 
} /* Output: 
222? 
Decoded using Cp1252: Some text 
Some text 
Some -text 
Ii~ 


缓冲 器 容纳 的 是 普通 的 字 节 ， 为 了 把 它们 转换 成 字符 ， 我 们 要 么 在 输入 它们 的 时 候 对 其 进 
行 编码 (这样 ， 它 们 输出 时 才 具 有 意义 )， 要 么 在 将 其 从 缓冲 器 输出 时 对 它们 进行 解码 。 可 以 使 
用 java.nio.charset.Charset 类 实现 这 些 功 能 ， 该 类 提供 了 把 数据 编码 成 多 种 不 同类 型 的 字符 集 的 


IR: 
//: io/AvailableCharSets. java 
// Displays Charsets and aliases 
import java.nio.charset.*; 
import java.util.*; 
import static net.mindview.util.Print.* 





public class AvailableCharsets { 
public static void main(String!) args) { 
SortedMap<String,Charset> charSets = 
Charset. availableCharsets() ; 
Iterator<String> it = charSets.keySet().iterator(); 
while(it.hasNext()) { 
String csName = it.next(); 
printnb(csName) ; 
Iterator aliases = 
charSets. get (csName) .aliases().iterator(); 
if (aliases. hasNext()) 
printnb(": "); 
while(aliases.hasNext()) { 
printnb(aliases.next()); 
if(aliases.hasNext()) 
printnb(", "); 








} 
printo; 
} 
} 
} 7* Output: 
Big5: csBigs 


BigS-HKSCS: bigS-hkscs, bigShk, big5-hkscs:unicode3.0, 
bigShkscs, Big5_HKSCS 

EUC-JP: eucjis, x-eucjp, csEUCPkdFmtjapanese, eucjp 
Extended_UNIX_Code_Packed_Format_for_Japanese, x-euc-jp, 
euc_jp 

EUC-KR: ksc5661，5661，ksc5691_1987，ksc_5691，ksc5661- 
1987, euc_kr, ks_¢_5601-1987, euckr, csEUCKR 

GB18030: gb18030-2600 

GB2312: gb2312-1980, gb2312, EUC_CN, gb2312-80, euc-cn, 























952] 





556. KISS 





euccn, x-EUC-CN 
GBK: Windows-936, CP936 


I~ 

让 我 们 返回 到 BufferToText.java， 如 果 我 们 想 对 缓冲 器 调用 rewind0 方 法 (调用 该 方法 是 为 
了 返回 到 数据 开始 部 分 )， 接 着 使 用 平台 的 黑 认 字符 集 对 数据 进行 decodeO ， 那 么 作为 结果 的 
CharBuffer 可 以 很 好 地 输出 打印 到 控制 台 。 可 以 使 用 System.getProperty("file.encoding") 发 现 默 
认 字 符 集 ， 它 会 产生 代表 字符 集 名 称 的 字符 串 。 把 该 字符 串 传送 给 CharsetforNaimne0 用 以 产生 
Charset 对 象 ， 可 以 用 它 对 字符 串 进行 解码 。 

另 一 选择 是 在 读 文件 时 ， 使 用 能 够 产生 可 打印 的 输出 的 字符 集 进行 encode()， 正 如 在 
BufferToText.java 中 第 3 部 分 所 看 到 的 那样 。 这 里 ，UTF-16BE 可 以 把 文本 写 到 文件 中 ， 当 读 取 
时 ， 我 们 只 需要 把 它 转换 成 CharBuffer， 就 会 产生 所 期 望 的 文本 。 

最 后 ， 让 我 们 来 看 看 若是 通过 CharBuffer 向 ByteBuffer 写 入 ， 会 发 生 什么 情况 (后 面 将 会 深 
入 了 解 ) 。 注 意 我 们 为 ByteBuffer 分 配 了 24 个 字 节 。 既 然 一 个 字符 需要 2 个 字 节 ， 那 么 一 个 
ByteBuffer 足 可 以 容纳 12 个 字符 ， 但 是 “Some text” 只 有 9 个 字符 ， 剩 余 的 内 容 为 零 的 字 节 仍 出 
现在 由 它 的 toString0 所 产生 的 CharBuffer 的 表示 中 ， 我 们 可 以 在 输出 中 看 到 。 

练习 23: (6) 创建 并 测试 一 个 实用 方法 ， 使 其 可 以 打印 出 CharBuffer 中 的 内 容 ， 直 到 字符 不 
能 再 打印 为 止 。 ` 


18.10.2 获取 基本 类 型 
尽管 ByteBuffer 只 能 保存 字 节 类 型 的 数据 ， 但 是 它 具 有 可 以 从 其 所 容纳 的 字 节 中 产生 出 各 种 
不 同 基本 类 型 值 的 方法 。 下 面 这 个 例子 展示 了 怎样 使 用 这 些 方法 来 插入 和 抽取 各 种 数值 : 


/1/: to/GetData, java 

// Getting different representations from a ByteBuffer 
‘import java.nio.*; 

import static net.mindview.util.Print.*; 


public class GetData { 
private static final int BSIZE = 1024; 
public static void main(String[] args) { 
ByteBuffer bb = ByteBuffer.allocate(BSIZE) ; 
// Allocation automatically zeroes the ByteBuffer: 
int 1 = 0; 
while(i++ < bb. Limit()) 
if(bb.get() != 9) 
print ("nonzero"); 
print("i =" + 4); 
bb-rewind() ; 
// Store and read a char array: 
bb. asCharBuffer() .put ("Howdy!"); 
char c; 
while((c = bb.getChar()) != 0) 
printnb(c + * "); 
print(); 
bb.rewind(): 
// Store and read a short: 
bb. asShortBuf fer () .put((short)471142); 
print(bb.getShort()): 
bb. rewind(); 
// Store and read an int: 
bb. asIntBuffer() .put (99471142) ; 
print(bb.getInt()); 
bb.rewind(); 
JJ Store and read a long: 
bb. asLongBuffer () . put (99471142) ; 





| 
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print(bb.getLong()); 

bb. rewind(); 

// Store and read a float: 

bb. asFloatBuffer() . put (99471142) ; 
print(bb.getFloat()); 

bb. rewind(); 

// Store and read a double: 

bb. asDoubleBuf fer () .put (99471142) ; 
print (bb. getDouble()); 

bb. rewind() ; 


* Output: 
= 1025 


99471142 
99471142 
9.9471144E7 
9.9471142E7 
Ii~ 


在 分 配 一 个 ByteBuffer 之 后 ， 可 以 通过 检测 它 的 值 来 查看 缓冲 器 的 分 配方 式 是 否 将 其 内 容 自 
动 置 零 一 它 确实 是 这 样 做 了 。 这 里 一 共 检 测 了 1024 个 值 ( 由 缓冲 器 的 limit0 决 定 )， 并 且 所 有 
的 值 都 是 零 。 

向 ByteBuffer 插 和 人 基本 类 型 数据 的 最 简单 的 方法 是 : 利用 asCharBuffer0、asShortBuffer0 等 
获得 该 缓冲 器 上 的 视图 ， 然 后 使 用 视图 的 put0 方 法 。 我 们 会 发 现 此 方法 适用 于 所 有 基本 数据 类 
型 。 仅 有 一 个 小 小 的 例外 ， 即 ， 使 用 ShortBuffer 的 put0 方 法 时 ， 需 要 进行 类 型 转换 (注意 类 型 
转换 会 截取 或 改变 结果 )。 而 其 他 所 有 的 视图 缓冲 器 在 使 用 put(0 方 法 时 ， 不 需要 进行 类 型 转换 。 


18.10.3 视图 缓冲 器 

视图 缓冲 器 (view buffer) 可 以 让 我 们 通过 某 个 特定 的 基本 数据 类 型 的 视窗 查看 其 底层 的 
ByteBuffer。ByteBuffer 依 然 是 实际 存储 数据 的 地 方 , “支持 ”着 前 面 的 视图 ， 因 此 ， 对 视图 的 
任何 修改 都 会 映射 成 为 对 ByteBuffer 中 数据 的 修改 。 正 如 我 们 在 上 一 示例 看 到 的 那样 ， 这 使 我 们 
可 以 很 方便 地 向 ByteBuffer 插 入 数据 。 视 图 还 允许 我 们 从 ByteBuffer 一 次 一 个 地 (与 ByteBuffer 
所 支持 的 方式 相同 ) 或 者 成 批 地 ( 放 入 数组 中 ) 读 取 基 本 类 型 值 。 在 下 面 这 个 例子 中 ， 通 过 
IntBuffer 操 纵 ByteBuffer 中 的 int 型 数据 : 


//: 40/IntBufferDemo. java 
// Manipulating ints in a ByteBuffer with an IntBuffer 
import java.nio.*; 


public class IntBufferDemo { 
Private static final int BSIZE = 1624; 
Public static void main(String{] args) { 
ByteBuffer bb = ByteBuffer.allocate(BSIZE); ° 
IntBuffer ib = bb.asIntBuffer(); 
// Store an array of int: 
ib.put(new int[]{ 11, 42, 47, 99, 143, 811, 1016 }); 
// Absolute location read and write: 
System. out.printin(ib.get(3)); 
ib.put(3, 1811); 
//, Setting a new Limit before rewinding the buffer. 
ib.ftipO; 
while(ib.hasRemaining()) { 
int 1 = fb.get(); 
System.out.printin(i); 
ti 


} 
} /* Output: 
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先 用 重 载 后 的 put() 方 法 存储 一 个 整数 数组 。 接 着 get0 和 put() 方 法 调用 直接 访问 底层 
ByteBuffer 中 的 某 个 整数 位 置 。 注 意 ， 这 些 通过 直接 与 ByteBuffer 对 话 访问 绝对 位 置 的 方式 也 同 
样 适 用 于 基本 类 型 。 

一 旦 底层 的 ByteBuffer 通 过 视图 缓冲 器 填 满 了 整数 或 其 他 基本 类 型 时 ， 就 可 以 直接 被 写 到 通 
道中 了 。 正 像 从 通道 中 读 取 那 样 容易 ， 然 后 使 用 视图 缓冲 器 可 以 把 任何 数据 都 转化 成 某 一 特定 
的 基本 类 型 。 在 下 面 的 例子 中 ， 通 过 在 同一 个 ByteBuffer 上 建立 不 同 的 视图 缓冲 器 ， 将 同一 字 节 
序列 翻译 成 了 short、int、float、long 和 double 类 型 的 数据 。 


//: io/ViewBuffers, java 
import java.nio.*; 
import static net.mindview.util.Print.*; 


public class ViewBuffers { 
public static void main(String[] args) { 
ByteBuffer bb = ByteBuffer.wrap( 
new byte(}{ ©. 0, 6, 0, 6, @, 8, ‘a’ P; 
bb. rewind(); 
printnb("Byte Buffer "); 
while (bb. hasRemaining()) 
printnb(bb.position()+ " -> " + bb.get() +", "); 
print(); 
CharBuffer cb = 
((ByteBuf fer) bb. rewind()) .asCharBuffer(); 
printnb("Char Buffer "): 
whi le (cb. hasRemaining()) 
printnb(cb.position() +" -> * + cb.get() +". "); 
print(); 
FloatBuffer fb = 
((ByteBuf fer) bb. rewind()) .asFloatBuffer(); 
printnb("Float Buffer "); 
while(fb.hasRemaining()) 
printnb(fb.position()+ " -> " + fb.get() +", 
print(); 
IntBuffer ib = 
((ByteBuffer)bb. rewind()).asIntBuffer(); 
printnb("Int Buffer "): 
while (ib. hasRemaining()) 
printnb({b.position()+ " -> " + ib.get() +", "); 
print(); æ 
LongBuffer tb = 
((ByteBuf fer) bb. rewind()) .asLongBuf fer () ; 
printnb("Long Buffer "); 
while(1b.hasRemaining()) 
printnb(1b.position()+ * -> " + lb.get() +", 
print(); 
ShortBuffer sb = 
((ByteBuf fer) bb. rewind()) .asShortBuf fer () ; 
printnb("Short Buffer "); 
while(sb.hasRemaining()) 
printnb(sb.position()+ “ -> " + sb.get() + ", 
print(); 
DoubleBuffer db = 
((ByteBuffer) bb. rewind()) .asDoubleBuffer(); 
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printnb("Double Buffer "); 
while (db.hasRemaining()) 
printnb(db.position()+ " -> ”+ db.get() +", "); 
} 

} /* Output: 
Byte Buffer © -> 9, 1 -> 0, 2 -> 6, 3 -> 6, 4 -> 0, 5 -> 0, 
6 -> 0, 7 -> 97, 
Char Buffer © -> ,1-> ,2-> ,3->a, 
Float Buffer @ -> 0.0, 1 -> 1.36E-43, 
Int Buffer @ -> @, 1 -> 97, 
Long Buffer @ -> 97, 
Short Buffer @ -> 6, 1 -> 6, 2 -> 6, 3 -> 97, 
Double Buffer © -> 4.8E-322, 
AAA 


ByteBuffer 通 过 一 个 被 “包装 ”过 的 8 字 节 数组 产生 ， 然 后 通过 各 种 不 同 的 基本 类 型 的 视图 
缓冲 器 显示 了 出 来 。 我 们 可 以 在 下 图 中 看 到 ， 当 从 不 同类 型 的 缓冲 器 读 取 时 ， 数 据 显 示 的 方式 
也 不 同 。 这 与 上 面 程序 的 输出 相对 应 。 


[ee o|o o | of © | sw Joves 


























4.8E-322 doubles 








练习 24，(1) 将 IntBufferDemojava 修 改 为 使 用 double。 

字 节 存放 次 序 

不 同 的 机 器 可 能 会 使 用 不 同 的 字 节 排序 方法 来 存储 数据 。“big endian” (高 位 优先 ) 将 最 重 
要 的 字 节 存 放 在 地 址 最 低 的 存储 器 单元 。 而 “little endian” (低位 优先 ) 则 是 将 最 重要 的 字 节 放 
在 地 址 最 高 的 存储 器 单元 。 当 存储 量 大 于 一 个 字 节 时 ， 像 int、float 等 ， 就 要 考虑 字 节 的 顺序 问 
题 了 。ByteBuffer 是 以 高 位 优先 的 形式 存储 数据 的 ， 并 且 数 据 在 网 上 传送 时 也 常常 使 用 高 位 优先 
的 形式 。 我 们 可 以 使 用 带 有 参数 ByteOrderBIG_ENDIAN 或 ByteOrderLITTLE_ENDIAN 的 
order0 方 法 改变 ByteBuffer 的 字 节 排序 方式 。 

考虑 包含 下 面 两 个 字 节 的 ByteBuffer: 


poopoooooopoggoggk| 
b1 b2 


如 果 我 们 以 short (ByteBuffer.asShortBuffer0) 形式 读 取 数 据 ， 得 到 的 数字 是 97 (二 进 制 形式 为 
00000000 01100001) ， 但 是 如 果 将 ByteBuffer 更 改 成 低位 优先 形式 ， 仍 以 short 形 式 读 取 数据 ， 
得 到 的 数字 却 是 24832 (二 进 制 形 式 为 01100001 00000000) 。 

这 个 例子 展示 了 怎样 通过 字 节 存放 模式 设置 来 改变 字符 中 的 字 节 次 序 : 


//: io/Endians.java 
// Endian differences and data storage. 
import java.nio.*; 
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import java.util.*; 
import static net.mindview.util.Print.*; 


public class Endians { 
public static void main(String[] args) { 

ByteBuffer bb = ByteBuffer.wrap(new byte[12]); 
bb. asCharBuffer() . put ("abcdef") ; 
print (Arrays. toString(bb.array())); 
bb. rewind(); 
bb. order (ByteOrder .BIG_ENDIAN) ; 
bb. asCharBuffer().put ("abcdef") ; 
print(Arrays.toString(bb.array())); 
bb. rewind(); 
bb. order (ByteOrder .LITTLE_ENDIAN) ; 
bb.asCharBuffer ().put ("abcdef") ; 
print (Arrays. toString(bb.array())); 


} 
} /* Output: 
(0, 97, ©, 98, 6, 99, @, 100, 6, 101, 6, 102] 
[0, 97, ©, 98. ©, 99, @, 160, 9, 101, ©, 102] 
[97, ©. 98, ©, 99, @, 100, ©, 101, 9, 102, 6) 
Whim 


ByteBuffer 有 足够 的 空间 ， 以 存储 作为 外 部 缓冲 器 的 charArray 中 的 所 有 字 节 ， 因 此 可 以 调 
用 array0 方 法 显示 视图 底层 的 字 节 。array0 方 法 是 “可 选 的 "， 并 且 我 们 只 能 对 由 数组 支持 的 组 
冲 器 调用 此 方法 ， 否 则 ， 将 会 抛 出 UnsupportedOperationException。 

通过 CharBuffer 视 图 可 以 将 charArray 插 入 到 ByteBuffer 中 。 在 底层 的 字 节 被 显示 时 ， 我 们 
会 发 现 默认 次 序 和 随后 的 高 位 优先 次 序 相同 ， 然 而 低位 优先 次 序 则 与 之 相反 ， 后 者 交换 了 这 些 
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字 节 次 序 。 
18.10.4 用 缓冲 器 操纵 数据 

下 面 的 图 六 明了 nio 类 之 间 的 关系 ， 便 于 我 们 理解 怎么 移动 和 转换 数据 。 例 如 ， 如 果 想 把 一 
个 字 节 数组 写 到 文件 中 去 ， 那 么 就 应 该 使 用 ByteBuffer.wrap0 方 法 把 字 节 数组 包装 起 来 ， 然 后 
用 getChannel0 方 法 在 FileOutputStream 上 打开 一 个 通道 ， 接 着 将 来 自 于 ByteBuffer 的 数据 写 到 
FileChannel 中 (如 下 页 图 所 示 )。 

注意 ，ByteBuffer 是 将 数据 移 进 移出 通道 的 唯一 方式 ， 并 且 我 们 只 能 创建 一 个 独立 的 基本 类 
型 缓冲 器 ， 或 者 使 用 “as” 方 法 从 ByteBuffer 中 获得 。 也 就 是 说 ， 我 们 不 能 把 基本 类 型 的 缓冲 器 
转换 成 ByteBuffer。 然 而 ， 由 于 我 们 可 以 经 由 视图 缓冲 器 将 基本 类 型 数据 移 进 移出 ByteBuffer， 
所 以 这 也 就 不 是 什么 真正 的 限制 了 。 
18.10.5 缓冲 器 的 细节 

Buffer 由 数据 和 可 以 高 效 地 访问 及 操纵 这 些 数据 的 四 个 索引 组 成 ， 这 四 个 索引 是 : mark 
(标记 ) position (位 置 )，limit (界限 ) 和 capacity (容量 )。 下 面 是 用 于 设置 和 复位 索引 以 及 查 
询 它 们 的 值 的 方法 。 


capacity) 返回 缓冲 区 容量 

dear) PERM, Hiposiionit RAO, limiit RASE. RITARA AEE 
fipo 将 limit 设 置 为 position , position 设置 为 0。 此 方法 用 于 准备 从 缓冲 区 读 取 已 经 写 人 的 数据 
limit, 返回 limit 值 

limit(int lim) 设置 limit 值 

markO 将 mark 设 置 为 position 

position) 返回 position 值 

Position(int pos) 设置 position 值 

remaining) 返回 (limit — position) 


hasRemainingO 车 有 介 于 position 和 limit 之 间 的 元 素 ， 则 返回 true 
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在 缓冲 器 中 插入 和 提取 数 : 引 ， 用 于 反映 所 发 生 的 变化 。 

下 面 的 示例 用 到 一 个 很 简单 的 算法 〈 交 换 相 邻 字符 )， 以 对 CharBuffer 中 的 字符 进行 编码 
(scramble) 和 译 码 (unscramble) 。 

1/: io/UsingBuffers.java 图 

import java.nio.* 1962 


import static net.mindview.util.Print.*; 








public class UsingBuffers ( 
private static void symmetricScramble(CharBuffer buffer) { 

while(buffer.hasRemaining()) { 

buffer.mark(); 

char cl = buffer.get(): 

char c2 = buffer.get(); 

buffer.reset(): 

buffer. put (c2).put (cl): 
} 





} 

public static void main(String[] args) { 
char{] data = "Usingbuffers”.toCharArray(); 
ByteBuffer bb = ByteBuffer.allocate(data.length * 2); 
CharBuffer cb = bb.asCharBuffer(): 
cb. put (data); 
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print(cb. rewind()); 
symmetricScramble(cb) ; 
print(cb.rewind()): 
symmetricScramble (cb); 
print(cb.rewind()); 


} 
} /* Output: 
UsingBuffers 
sUniBgfuefsr 
UsingBuffers 
Wh 


尽管 可 以 通过 对 某 个 char 数 组 调用 wrap0 方 法 来 直接 产生 一 个 CharBuffer， 但 是 在 本 例 中 
取而代之 的 是 分 配 一 个 底层 的 ByteBuffer, 产生 的 CharBuffer 只 是 ByteBuffer 上 的 一 个 视图 而 已 。 
这 里 要 强调 的 是 ， 我 们 总 是 以 操纵 ByteBuffer 为 目标 ， 因 为 它 可 以 和 通道 进行 交互 。 

下 面 是 进入 symmetrieSeramble0 方 法 时 缓冲 器 的 样子 : 


ppogpogpogdot 


Position 指针 指向 缓冲 器 中 的 第 一 个 元 素 ，capacity 和 limit 则 指向 最 后 一 个 元 素 。 

在 程序 的 symmetricScramble0 方 法 中 ， 和 迭代 执行 while 循 环 直到 position 等 于 limit。 一 旦 调用 
缓冲 器 上 相对 的 get0 或 put0 函 数 ，position 指 针 就 会 随 之 相应 改变 。 我 们 也 可 以 调用 绝对 的 、 包 
含 一 个 索引 参数 的 get0 和 put0 方 法 (参数 指明 get0 或 pat0 的 发 生 位 置 )。 不 过 ， 这 些 方法 不 会 改 
变 缓冲 器 的 position 指 针 。 

当 操纵 到 while 循 环 时 ， 使 用 markO 调 用 来 设置 mark 的 值 。 此 时 ， 缓 冲 器 状态 如 下 ， 





pag 


om oa 
LELI TTT] 
= om 


两 个 相对 的 get( 调 用 把 前 两 个 字符 保存 到 变量 cl 和 e2 中 ， 调 用 完 这 两 个 方法 后 ， 缓 冲 器 
如 下 
(eraj Go 


为 了 实现 交换 ， 我 们 要 在 position=0 时 写 人 ec2，position = 1 时 写 和 el。 我 们 也 可 以 使 用 绝对 


的 put0 方 法 来 实现 ， 或 者 使 用 reset0 把 position 的 值 设 为 mark 的 值 : 


mo CE 


Ga 


这 两 个 put0 方 法 先 写 c2， 接 着 写 cl: 
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T CE 
CEDE Apo TEH 
[ral Gn 

在 下 一 次 循环 迭代 期 间 ， 将 mark 设 置 成 position 的 当前 值 ; 
Gao eo 
GE r : 
GE [uml 


这 个 过 程 将 会 持续 到 遍历 完整 个 缓冲 器 。 在 while 循 环 的 最 后 ，position 指 向 缓冲 器 的 末尾 。 
如 果 要 打印 缓冲 器 ， 只 能 打印 出 position 和 limit 之 间 的 字符 。 因 此 ， 如 果 想 显示 缓冲 器 的 全 部 内 
容 ， 必 须 使 用 rewind0 把 position 设 置 到 缓冲 器 的 开始 位 置 。 下 面 是 调用 rewind(0 之 后 缓冲 器 的 状 
AS (mark 的 值 则 变 得 不 明确 ): 


mpm 


当 再 次 调用 symmetricScramble0 功 能 时 ， 会 对 CharBuffer 进 行 同样 的 处 理 ， 并 将 其 恢复 到 
初始 状态 。 
18.10.6 内 存 映 射 文件 

内 存 映 射 文件 允许 我 们 创建 和 修改 那些 因为 太 大 而 不 能 放 入 内 存 的 文件 。 有 了 内 存 映射 文 
件 ， 我 们 就 可 以 假定 整个 文件 都 放 在 内 存 中 ， 而 且 可 以 完全 把 它 当 作 非 常 大 的 数组 来 访问 。 这 
种 方法 极 大 地 简化 了 用 于 修改 文件 的 代码 。 下 面 是 一 个 小 例子 : 


//: \o/LargeħappedFi les. java 

// Creating a very large file using mapping. 
/1 {RunByHand) 

import java.nio.*; 

import java.nio.channels.*; 

import java. o.*; 

import static net.mindview.util.Print.*; 





public class LargeMappedFiles { 
static int length = @x8FFFFFF; // 128 MB 
public static void main(String{] args) throws Exception { 
MappedByteBuffer out = 
new RandomAccessFile("test.dat", "rw").getChannel() 
-map (Fi leChannel .MapMode .READ_WRITE, @, length); 
for(int 1 = @; 1 < length; i++) 
out. put ((byte)'x'); 
print("Finished writing"); 
for(int 1 = length/2; 1 < length/2 + 6; i++) 
printnb((char)out.get(i)); 


} Mi~ 
为 了 既 能 写 又 能 读 ， 我 们 先 由 RandomAccessFile 开 始 ， 获 得 该 文件 上 的 通道 ， 然 后 调用 


map0 产 生 MappedByteBuffer， 这 是 一 种 特殊 类 型 的 直接 缓冲 器 。 注 意 我 们 必须 指定 映射 文件 
的 初始 位 置 和 映射 区 域 的 长 度 ， 这 意味 着 我 们 可 以 映射 某 个 大 文件 的 较 小 的 部 分 。 
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MappedByteBuffer 由 ByteBuffer 继 承 而 来 ， 因 此 它 具 有 ByteBuffer 的 所 有 方法 。 这 里 ， 我 们 
仅仅 展示 了 非常 简单 的 put0 和 get0， 但 是 我 们 同样 可 以 使 用 像 sCharBuffer0 等 这 样 的 用 法 。 

前 面 那 个 程序 创建 的 文件 为 128MB， 这 可 能 比 操作 系统 所 允许 一 次 载 和 内存 的 空间 大 。 但 
似乎 我 们 可 以 一 次 访问 到 整个 文件 ， 因 为 只 有 一 部 分 文件 放 入 了 内 存 ， 文 件 的 其 他 部 分 被 交换 
了 出 去 。 用 这 种 方式 ， 很 大 的 文件 (可 达 2GB) 也 可 以 很 容易 地 修改 。 注 意 底层 操作 系统 的 文 
件 映 射 工具 是 用 来 最 大 化 地 提高 性 能 。 

性 能 

尽管 “ 旧 ” 的 VO 流 在 用 nio 实 现 后 性 能 有 所 提高 ， 但 是 “映射 文件 访问 ”往往 可 以 更 加 显著 
地 加 快速 度 。 下 面 的 程序 进行 了 简单 的 性 能 比较 。 


//: io/MappedI0.java 
import java.nio.*; 
import java.nio.channels.*; 
import java.io.*; 


public class MappedIO { 
private static int numOfInts = 4000000; 
Private static int numOfUbuffInts = 200000; 
Private abstract static class Tester { 
private String name; 
public Tester (String name) { this.name = name; } 
public void runTest() { 
System.out.print(name + ": 
try { 
long start = System.nanoTime(); 
test(); 
double duration = System.nanoTime() - start; 
System. out. format("%.2f\n", duration/1.0e9): 
catch (IOException e) { 
throw new Runtime€xception(e) ; 
} 


} 
public abstract void test() throws IOException: 





) 


} 
private static Tester[] tests = { 
new Tester("Stream Write") { 
public void test() throws IOException { 
DataOutputStream dos = new DataQutputStream( 
new BufferedOutputStream( 
new FileOutputStream(new File("temp.tmp")))); 
for(int i = 0; 1 < numOfInts; i++) 
dos.writeInt(i); 
dos.close(); 


} 
}, 
new Tester ("Mapped Write") { 
public void test() throws IOException { 

FileChannel fc = 
new RandomAccessFile("temp.tmp", “rw") 
-getChannel(); 

IntBuffer ib = fc.map( 
FileChannel.MapHode.READ WRITE, ©, fc.size()) 
.asIntBuffer(); 

for(int i = @; i < numOfInts; i++) 
ib. puti 

fe.close(); 

} 








}, 
new Tester("Stream Read") { 
public void test() throws IOException { 
DataInputStream dis = new DataInputStream( 
new Buf feredInputStream( 
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new FileInputStream("temp.tmp"))); 
for(int 1 = @; 1 < numOfInts: i++) 
dis.readInt(); 
dis.close(); 
} 
}, 
new Tester("Mapped Read") { 
public void test() throws IOException { 
FileChannel fc = new FileInputStream( 
new File("temp.tmp")).getChannel(); 
IntBuffer ib = fc.map( 
FileChannel.MapMode.READ_ONLY, ©, fc.size()) 
-asIntBuffer(); 
white(ib.hasRemaining()) 
ib.get( 
fe.close(); 
} 
a 
new Tester ("Stream Read/Write") { 
public void test() throws IOException { 
RandomAccessFile raf = new RandomAccessFile( 
new File("temp.tmp"), "rw"); 
raf .writeInt(1); 
for(int i = 8; 1 < numOfUbuffInts; i++) { 
raf.seek(raf.length() - 4); 
raf.weitelnt(raf.readint()); 








} 
raf.close(); 
} 
}, 
new Tester ("Mapped Read/Write") { 
public void test() throws IOException { 
FileChannel fc = new RandomAccessFile( 
new File("temp. tmp” 本 "rw") .getChannet(); 
IntBuffer ib = fc.map' 
FileChannel. Haptode READ_WRITE, 6, fc.size()) 
-asIntBuffer(); 
ib. put(@); 
for(int 1 = 1; 1 < numOfUbuffints; i++) 
ib.put(ib.get(i - 1)); 
fe.close(); 
} 
} 


ye 
public static void main(String(] args) { 
for(Tester test : tests) 
test.runTest(); 





} 

} /* Output: (96% match) 
Stream Write: @.56 
Mapped Write: 0.12 
Stream Read: 0.80 
Mapped Read: 0.07 
Stream Read/Write: 5.32 
Mapped Read/Write: 0.02 
Wm 


正如 在 本 书 前 面 的 例子 中 所 看 到 的 那样 ，runTest0 被 用 作 是 一 种 模板 方法 ， 为 在 匿名 内 部 
子 类 中 定义 的 test0 的 各 种 实现 创建 了 测试 框架 )。 每 种 子 类 都 将 执行 一 种 测试 ， 因 此 test0 方 法 
为 我 们 进行 各 种 /O 操 作 提供 了 原型 。 
i 尽管 “映射 写 ”似乎 要 用 到 FileOutputStream， 但 是 映射 文件 中 的 所 有 输出 必须 使 用 
t RandomAccessFile， 正 如 前 面 程序 代码 中 的 读 / 写 一 样 。 [969] . 
注意 test0 方 法 包括 初始 化 各 种 MO 对 象 的 时 间 ， 因 此 ， 即 使 建立 映射 文件 的 花费 很 大 ， 但 是 
整体 受益 比 起 MO 流 来 说 还 是 很 显著 的 。 
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练习 25: (6) 试 着 将 本 章 例子 中 的 ByteBuffer.allocate0 语 句 改 为 ByteBuffer.allocateDirect0。 
用 来 证 实 性 能 之 间 的 差异 ， 但 是 请 注意 程序 的 启动 时 间 是 否 发 生 了 明显 的 改变 。 

练习 26: (3) 修改 JGrep.java， 让 其 使 用 Java 的 nio 内 存 映 射 文件 。 
18.10.7 文件 加 锁 

IDK 1.4 引 入 了 文件 加 锁 机 制 ， 它 允许 我 们 同步 访问 某 个 作为 共享 资源 的 文件 。 不 过 ， 竞 争 
同一 文件 的 两 个 线程 可 能 在 不 同 的 Java 虚 拟 机 上 ， 或 者 一 个 是 Java 线 程 ， 另 一 个 是 操作 系统 中 其 
他 的 某 个 本 地 线程 。 文 件 锁 对 其 他 的 操作 系统 进程 是 可 见 的 ， 因 为 Java 的 文件 加 锁 直 接 映射 到 
了 本 地 操作 系统 的 加 锁 工具 。 

下 面 是 一 个 关于 文件 加 锁 的 简单 例子 。 


11: to/FileLocking.java 
import java.nio.channels.*; 
import java.utit.concurrent.*; 
import java.to.*; 


public class FileLocking { 
public static void main(String{] args) throws Exception { 

FileOutputStream fos= new FileOutputStream("file. txt"); 
FileLock fl = fos.getChannel().tryLock() ; 
if(fl != null) { 

System.out.printin("Locked File"); 

TimeUnit MILLISECONDS. sleep (100) ; 

fl.release(); 

System.out.printin("Released Lock"); 


} 
fos.close(); 


} 
} /* Output: 
Locked File 
Released Lock 
Whim 


通过 对 FileChannel 调 用 tryLock0) 或 lockO， 就 可 以 获得 整个 文件 的 FileLock。(Socket- 
Channel、DatagramChannel 和 ServerSocketChannel 不 需要 加 锁 ， 因 为 它们 是 从 单 进程 实体 继 
承 而 来 ， 我们 通常 不 在 两 个 进程 之 间 共 享 网 络 socket。 ) tryLock0 是 非 阻塞 式 的 ， 它 设法 获取 锁 ， 
但 是 如 果 不 能 获得 〈 当 其 他 一 些 进程 已 经 持 有 相同 的 锁 ， 并 且 不 共享 时 ) ， 它 将 直接 从 方法 调用 
返回 。lock0 则 是 阻塞 式 的 ， 它 要 阻塞 进程 直至 锁 可 以 获得 ， 或 调用 lock0 的 线程 中 断 ， 或 调用 
lock0 的 通道 关闭 。 使 用 FileLock.release0 可 以 释放 镇 。 

也 可 以 使 用 如 下 方法 对 文件 的 一 部 分 上 锁 : 

tryLock(long position, long size, boolean shared) 
或 者 

lock(long position, long size, boolean shared) 
其 中 ， 加 锁 的 区 域 由 size-position 决 定 。 第 三 个 参数 指定 是 否 是 共享 锁 。 

尽管 无 参数 的 加 锁 方法 将 根据 文件 尺寸 的 变化 而 变化 ， 但 是 具有 固定 尺寸 的 锁 不 随 文件 尺 
寸 的 变化 而 变化 。 如 果 你 获得 了 某 一 区 域 (从 position 到 position + size) 上 的 锁 ， 当 文件 增 大 超 
出 position+size 时 ， 那 么 在 Position+size 之 外 的 部 分 不 会 被 锁定 。 无 参数 的 加 锁 方法 会 对 整个 文件 
进行 加 锁 ， 甚 至 文件 变 大 后 也 是 如 此 。 

对 独占 锁 或 者 共享 锁 的 支持 必须 由 底层 的 操作 系统 提供 。 如 果 操 作 系统 不 支持 共享 锁 并 为 
每 一 个 请 求 都 创建 一 个 锁 ， 那 么 它 就 会 使 用 独占 锁 。 锁 的 类 型 (共享 或 独占 ) 可 以 通过 
FileLock.isShared0 进行 查询 。 
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对 映射 文件 的 部 分 加 锁 
如 前 所 述 ， 文 件 映射 通常 应 用 于 极 大 的 文件 。 我 们 可 能 需要 对 这 种 巨大 的 文件 进行 部 分 加 
锁 ， 以 便 其 他 进程 可 以 修改 文件 中 未 被 加 锁 的 部 分 。 例 如 ， 数 据 库 就 是 这 样 ， 因 此 多 个 用 户 可 
以 同时 访问 到 它 。 
下 面 例子 中 有 两 个 线程 ， 分 别 加 锁 文 件 的 不 同 部 分 。 ical 


//: io/LockingMappedFiles. java 
// Locking portions of a mapped file. 
// {RunByHand} 

import java.nio.*; 

import java.nio.channels.*; 

import java.io.*; 





public class LockingMappedFiles { 
static final int LENGTH = @xBFFFFFF; // 128 MB 
static FileChannel fc; 
public static void main(String[{] args) throws Exception { 
fc = 
new RandomAccessFile("test.dat", "rw").getChannel(); 
MappedByteBuffer out = 
fc.map(FileChannel.MapMode.REAO_WRITE, @, LENGTH); 
for(int i = 0; 1 < LENGTH: i++) 
out. put( (byte) 'x'); 
new LockAndModify(out, @, © + LENGTH/3); 
new LockAndModify(out, LENGTH/2, LENGTH/2 + LENGTH/4); 
} 
private static class LockAndModify extends Thread { 
private ByteBuffer buff; 
private int start, end; 
LockAndModify(ByteBuffer mbb, int start, int end) { 
this.start = start; 
this.end = end; 
mbb. Limit (end) ; 
mbb.position(start); 
buff = mbb.slice(); 
start(); 


public void run() { 
try { 
// Exclusive lock with no overlap: 
FileLock fl = fc.lock(start, end, false); 
System.out.printin(*Locked: "+ start +" to "+ end); ` 
// Perform modification: 
while(buff .position() < buff.limit() - 1) 
buff.put( (byte) (buff.get() + 1)); 
fl.retease(): 
System. out.printin("Released: "+start+" to “+ end); 
catch (IOException e) { 
throw new RuntimeException(e); 


} [972 
} 
} 

dim 

线程 类 LockAndModify 创建 了 缓冲 区 和 用 于 修改 的 slice0， 然 后 在 run0 中 ， 获 得 文件 通道 
上 的 锁 (我们 不 能 获得 缓冲 器 上 的 锁 ， 只 能 是 通道 上 的 ) 。lockO 调 用 类 似 于 获得 一 个 对 象 的 线 
程 锁 一 -我 们 现在 处 在 “临界 区 ”， 即 对 该 部 分 的 文件 具有 独占 访问 权 。。 

如 果 有 Java 虚 拟 机 ， 它 会 自动 释放 锁 ， 或 者 关闭 加 锁 的 通道 。 不 过 我 们 也 可 以 像 程序 中 那 
样 ， 显 式 地 为 FileLock 对 象 调用 release0 来 释放 锁 。 














日 有 关 线 程 的 更 多 细节 在 第 21 章 中 可 以 找到 。 
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18.11 压缩 


Java VO 类 库 中 的 类 支持 读 写 压缩 格式 的 数据 流 。 你 可 以 用 它们 对 其 他 的 VO 类 进行 封装 ， 以 
提供 压缩 功能 。 

这 些 类 不 是 从 Reader 和 Writer 类 派生 而 来 的 ， 而 是 属于 InputStream 和 OutputStream 继 承 层 次 
结构 的 一 部 分 。 这 样 做 是 因为 压缩 类 库 是 按 字 节 方式 而 不 是 字符 方式 处 理 的 。 不 过 有 时 我 们 可 能 会 
被 迫 要 混合 使 用 两 种 类 型 的 数据 流 (注意 我 们 可 以 使 用 mputStreamReader 和 Output-StreamWriter 























在 两 种 类 型 间 方 便 地 进行 转换 )。 
压 编 类 功 能 
CheckedInputStream GetCheckSum() 为 任何 InputStream 产 生 校 验 和 (不 仅 是 解压 缩 ) 
CheckedOutputStream GetCheckSum( ) 为 任何 OutputStream 产 生 校 验 和 (不 仅 是 压缩 ) 
CE DeflaterOutputStream 压缩 类 的 基 关 
ZipOutputStream 一 个 DeflaterOutputStream ， 用 于 将 数据 压缩 成 Zip 文件 格式 
GZIPOutputStream 一 个 DeflaterOutputStream ， 用 于 将 数据 压缩 成 GZIP 文件 格式 
InflaterinputStream 解压 缩 类 的 基 类 
ZipInputStream 一 个 InflaterInputStream， 用 于 解压 缩 Zip 文 件 格式 的 数据 
GZIPInputStream 一 个 IaflaterInputStream， 用 于 解压 缩 GZIP 文 | 的 数据 





尽管 存在 许多 种 压缩 算法 ， 但 是 Zip 和 GZIP 可 能 是 最 常用 的 。 因 此 我 们 可 以 很 容易 地 使 用 多 
种 可 读 写 这 些 格式 的 工具 来 操纵 我 们 的 压缩 数据 。 
18.11.1 用 GZIP 进 行 简单 压缩 

GZIP 接 口 非常 简单 ， 因 此 如 果 我 们 只 想 对 单个 数据 流 (而 不 是 一 系列 互 异 数据 ) 进行 压缩 ， 
那么 它 可 能 是 比较 适合 的 选择 。 下 面 是 对 单个 文件 进行 压缩 的 例子 : 


/1/: io/GZIPcompress.java 
// {Args: G2IPcompress.java} 
import java.util.zip.*; 
import java.io.*; 


public class GZIPcompress { 
public static void main(Stringl] args) 
throws IOException { 
if(args.length == 6) ( 
System. out.printin( 
“Usage: \nGZIPcompress file\n" + 
"\tUses GZIP compression to compress * + 
"the file to test.gz"); 
System.exit(1); 


BufferedReader in = new BufferedReader ( 
new FileReader (args[@])): 
BufferedQutputStream out = new BufferedOutputStream( 
974} new GZIPOutputStream( 
new FileOutputStream("test.gz"))); 
System.out.printin("Writing file"); 
int c; 
while((c = in.read()) != -1) 
out.write(c); 
in.close(); 
out.close(); 
System.out.printIn("Reading file"); 
BufferedReader in2 = new BufferedReader ( 
new InputStreamReader (new GZIPInputStream( 
new FileInputStream("test.gz")))); 
| String s; 
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while((s = in2.readLine()) != null) 
System.out.printin(s); 


} A (Execute to see output) *///:~ 

压缩 类 的 使 用 非常 直观 一 直接 将 输出 流 封装 成 GZIPOutputStream 或 ZipOutputStream， 并 将 
输入 流 封装 成 GZIPInputStream 或 ZipInputStream 即 可 。 其 他 全 部 操作 就 是 通常 的 VO 读 写 。 这 个 例 
子 把 面向 字符 的 流 和 面向 字 节 的 流 混合 了 起 来 HA (in) 用 Reader 类 ， 而 GZIPOutputStream 的 
构造 器 只 能 接受 OutputStream 对 象 ， 不 能 接受 Writer 对 象 。 在 打开 文件 时 ，GZIPInputStream 就 会 
被 转换 成 Reader。 
18.11.2 用 Zip 进 行 多 文件 保存 

支持 Zip 格 式 的 Java 库 更 加 全 面 。 利 用 该 库 可 以 方便 地 保存 多 个 文件 ， 它 甚至 有 一 个 独立 的 
类 ， 使 得 读 取 Zip 文 件 更 加 方便 。 这 个 类 库 使 用 的 是 标准 Zip 格 式 ， 所 以 能 与 当前 那些 可 通过 因 
特 网 下 载 的 压缩 工具 很 好 地 协作 。 下 面 这 个 例子 具有 与 前 例 相同 的 形式 ， 但 它 能 根据 需要 来 处 
理 任意 多 个 命令 行 参数 。 另 外 ， 它 显示 了 用 Checksum 类 来 计算 和 校 验 文件 的 校 验 和 的 方法 。 一 
共有 两 种 Checksum 类 型 ，Adler32 ( 它 快 一 些 ) 和 CRC32 ( 慢 一 些 ， 但 更 准确 ) 。 


/1/: to/ZipCompress.java 
// Uses Zip compression to compress any 

// number of files given on the command line. 
// {Args: ZipCompress. java} 

import java.util.zip.*; 

import java.to.* 
import java.util.*; 

import static net.mindview.util.Print.*; 





public class ZipCompress { 
public static void main(String[] args) 
throws IOException { 
FileOutputStream f = new FileOutputStream("test.zip"); 
CheckedOutputStream csum = 
new CheckedOutputStream(f, new Adler32()); 
ZipOutputStream zos = new ZipOutputStream(csum) ; 
BufferedOutputStream out = 
new Buf feredOutputStream(zos) ; 
zos.setConment("A test of Java Zipping"); 
// No corresponding getComment(), though. 
for(String arg : args) { 
print("Writing file ”+ arg): 
BufferedReader in = 
new BufferedReader (new FileReader(arg)); 
zos.putNextEntry(new Zipentry(arg)); 
int c; 
while((c = in.read()) != -1) 
out .write(c); 
in.close(); 
out. flush(); 


} 
out.close(); 
// Checksum valid only after the file has been closed! 
print("Checksum: ”+ csum.getChecksum().getValue()); 
// Now extract the files: 
print(*Reading file"); 
FileinputStream fi = new FileInputStream("test.zip"); 
CheckedInputStream csumi = 

new CheckedInputStream(fi, new Adler32()): 
ZipInputStream in2 = new ZipInputStream(csumt): 
BufferedInputStream bis = new BufferedinputStream(in2); 
ZipEntry ze; 
while((ze = in2.getNextEntry()) != mull) { 
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print("Reading file " + ze); 

int x; 

while((x = bis.read()) != -1) 
System. out .write(x); 


} 
if(args.length == 1) 
print("Checksum: ”+ csumi.getChecksum().getValue()); 
bis.close(); 
// Alternative way to open and read Zip files: 
ZipFile zf = new ZipFile(*test.zip"); 
Enumeration e = zf.entries(); 
while(e.hasMoreElements()) { 
ZipEntry ze2 = (ZipEntry)e.nextElement(); 
print("File: "+ ze2); 
// ... and extract the data as before 
} 
/* if(args.length == 1) */ 
} 
} /* (Execute to see output) *///:~ 


对 于 每 一 个 要 加 入 压缩 档案 的 文件 ， 都 必须 调用 putNextEntry()， 并 将 其 传递 给 一 个 
ZipEntry 对 象 。ZipEntry 对 象 包含 了 一 个 功能 很 广泛 的 接口 ， 允 许 你 获取 和 设置 Zip 文 件 内 该 特 
定 项 上 所 有 可 利用 的 数据 : 名字、 压缩 的 和 未 压缩 的 文件 大 小 、 日 期 、CRC 校 验 和 、 额 外 字段 数 
据 、 注 释 、 压 缩 方法 以 及 它 是 否 是 一 个 目录 人 口 等 等 。 然 而 ， 尽 管 Zip 格 式 提供 了 设置 密码 的 方 
法 ， 但 Java 的 Zip 类 库 并 不 提供 这 方面 的 支持 。 虽 然 CheckedInputStream 和 CheckedOutputStream 
都 支持 Adler32 和 CRC32 两 种 类 型 的 校 验 和 ， 但 是 ZipEntry 类 只 有 一 个 支持 CRC 的 接口 。 虽 然 这 
是 一 个 底层 Zip 格 式 的 限制 ， 但 却 限制 了 人 们 不 能 使 用 速度 更 快 的 Adler32。 

为 了 能 够 解压 缩 文件 ，ZipInputStream 提 供 了 一 个 getNextEntry0 方 法 返回 下 一 个 ZipEntry 
(如 果 存 在 的 话 )。 解 压缩 文件 有 一 个 更 简便 的 方法 一 利用 ZipFile 对 象 读 取 文 件 。 该 对 象 有 一 
个 entries0 方 法 用 来 向 ZipEntries 返 回 一 个 Enumeration ( 枚 举 ) 。 

为 了 读 取 校 蛤 和 ， 必 须 拥有 对 与 之 相关 联 的 Checksum 对 象 的 访问 权限 。 在 这 里 保留 了 指向 
CheckedOutputStream 和 CheckedInputStream 对 象 的 引用 。 但 是 ， 也 可 以 只 保留 一 个 指向 
Checksum 对 象 的 引用 。 

Zip 流 中 有 一 个 令 人 困惑 的 方法 setComment0。 正 如 前 面 ZipCompressjava 中 所 示 ， 我 们 可 
以 在 写 文件 时 写 注释 , 但 却 没有 任何 方法 恢复 ZipInputStream 内 的 注释 。 似乎 只 能 通过 ZipEntry， 
才能 以 逐条 方式 完全 支持 注释 的 获取 。 

当然 ，GZIP 或 Zip 库 的 使 用 并 不 仅仅 局 限于 文件 一 它 可 以 压缩 任何 东西 ， 包 括 需要 通过 网 
络 发 送 的 数据 。 

18.11.3 Java 档 案 文件 

Zip 格 式 也 被 应 用 于 JAR (Java ARchive, Java 档 案 文件 ) 文件 格式 中 。 这 种 文件 格式 就 像 Zip 
一 样 ， 可 以 将 一 组 文件 压缩 到 单个 压缩 文件 中 。 同 Java 中 其 他 任何 东西 一 样 ，JAR 文 件 也 是 跨 平 
台 的 ， 所 以 不 必 担心 跨 平台 的 问题 。 声 音 和 图 像 文 件 可 以 像 类 文件 一 样 被 包含 在 其 中 。 

JAR 文 件 非常 有 用 ， 尤 其 是 在 涉及 因特网 应 用 的 时 候 。 如 果 不 采用 JAR 文 件 ，Web 浏 览 器 在 
下 载 构成 一 个 应 用 的 所 有 文件 时 必须 重复 多 次 请 求 Web 服 务 器 ， 而且 所 有 这 些 文件 都 是 未 经 压 
缩 的 。 如 果 将 所 有 这 些 文件 合并 到 一 个 JAR 文 件 中 ， 只 需 向 远程 服务 器 发 出 一 次 请 求 即 可 。 同 
时 ， 由 于 采用 了 压缩 技术 ， 可 以 使 传输 时 间 更 短 。 另 外 ， 出 于 安全 的 考虑 ，JAR 文 件 中 的 每 个 
条 目 都 可 以 加 上 数字 化 签名 。 

一 个 JAR 文 件 由 一 组 压缩 文件 构成 ， 同 时 还 有 一 张 描述 了 所 有 这 些 文件 的 “文件 清单 ”( 可 
自行 创建 文件 清单 ， 也 可 以 由 jar 程 序 自动 生成 )。 在 JDK 文 档 中 ， 可 以 找到 与 JAR 文 件 清单 相关 
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的 更 多 资料 。 

Sun 的 JDK 自 带 的 jar 程 序 可 根据 我 们 的 选择 自动 压缩 文件 。 可 以 用 命令 行 的 形式 调用 它 ， 如 
下 所 示 : 

jar [options] destination [manifest] inputfile(s) 
其 中 options 只 是 一 个 字母 集合 (不必 输入 任何 “-” 或 其 他 任何 标识 符 )。 以 下 这 些 选 项 字符 在 
Unix 和 Linux 系 统 中 的 tar 文 件 中 也 具有 相同 的 意义 。 具 体 意义 如 下 所 示 : 





c 创建 一 个 新 的 或 空 的 压缩 文档 

列 出 目录 表 
x 解压 所 有 文件 
x file 解压 该 文件 
f 意 指 :“ 我 打算 指定 一 个 文件 名 。” 如 果 没 有 用 这 个 选项 ，jar 假设 所 有 的 输入 都 来 自 于 标准 输入 ， 

或 者 在 创建 一 个 文件 时 ， 输 出 对 象 也 假设 为 标准 输出 978 

m 表示 第 一 个 参数 将 是 用 户 自 建 的 清单 文件 的 名 字 
v 产生 详细 和 输出， 描述 jar 所 做 的 工作 
o 只 储存 文件 ， 不 压缩 文件 (用 来 创建 一 个 可 放 在 类 路 径 中 的 JAR 文 件 ) 
M 不 自动 创建 文件 清单 





如 果 压 缩 到 JAR 文 件 的 众多 文件 中 包含 某 个 子 目 录 ， 那 么 该 子 目 录 会 被 自动 添加 到 JAR 文 件 
中 ， 且 包括 该 子 目录 的 所 有 子 目 录 ， 路 径 信息 也 会 被 保留 。 

以 下 是 一 些 调用 jar 的 典型 方法 。 下 面 的 命令 创建 了 一 个 名 为 myJarFilejar 的 JAR 文 件 ， 该 
文件 包含 了 当前 目录 中 的 所 有 类 文件 ， 以 及 自动 产生 的 清单 文件 : 

jar cf myJarFile.jar *.class 
下 面 的 命令 与 前 例 类 似 ， 但 添加 了 一 个 名 为 myManifestFile.mf 的 用 户 自 建 清单 文件 : 

jar cmf myJarFile.jar myManifestFile.mf *.class 
下 面 的 命令 会 产生 myJarFilejar 内 所 有 文件 的 一 个 目录 表 : 

jar tf myJarFile.jar 
下 面 的 命令 添加 “v”( 详 尽 ) 标志 ， 可 以 提供 有 关 myJarFilejar 中 的 文件 的 更 详细 的 信息 : 

jar tvf myJarFile.jar 
假定 audio、classes 和 image 是 子 目 录 ， 下 面 的 命令 将 所 有 子 目录 合并 到 文件 myAppjar 中 ， 其 中 
也 包括 了 “v” 标 志 。 当 jar 程 序 运行 时 ， 该 标志 可 以 提供 更 详细 的 信息 : 

jar cvf myApp.jar audio classes image 

如 果 用 0 (P) 选项 创建 一 个 JAR 文 件 ， 那 么 该 文件 就 可 放 入 类 路 径 变量 (CLASSPATH) 中 : 

CLASSPATH="1ibl.jar;lib2.jar:" 
然后 Java 就 可 以 在 libljar 和 lib2jar 中 搜索 目标 类 文件 了 。 

jar 工 具 的 功能 没有 zip 工 具 那 么 强大 。 例 如 ， 不 能 够 对 已 有 的 JAR 文 件 进行 添加 或 更 新 文件 
的 操作 ， 只 能 从 头 创 建 一 个 JAR 文 件 。 同 时 ， 也 不 能 将 文件 移动 至 一 个 JAR 文 件 ， 并 在 移动 后 将 
它们 删除 。 然 而 ， 在 一 种 平台 上 创建 的 JAR 文 件 可 以 被 在 其 他 任何 平台 上 的 jar 工 具 透 明 地 阅读 
(这 个 问题 有 时 会 困扰 zip 工 具 ) 。 

读者 将 会 在 第 22 章 看 到 ，JAR 文 件 也 被 用 来 为 JavaBeans 打 包 。 


18.12 对 象 序列 化 
当 你 创建 对 象 时 ， 只 要 你 需要 ， 它 就 会 一 直 存在 ， 但 是 在 程序 终止 时 ， 无 论 如何 它 都 不 会 


S] 
3| 
S| 


| iia i id 








572 HIS 





继续 存在 。 尽 管 这 么 做 肯定 是 有 意义 的 ， 但 是 仍旧 存在 某 些 情况 ， 如 果 对 象 能 够 在 程序 不 运行 
的 情况 下 仍 能 存在 并 保存 其 信息 ， 那 将 非常 有 用 。 这 样 ， 在 下 次 运行 程序 时 ， 该 对 象 将 被 重建 
并 且 拥有 的 信息 与 在 程序 上 次 运行 时 它 所 拥有 的 信息 相同 。 当 然 ， 你 可 以 通过 将 信息 写 入 文件 
或 数据 库 来 达到 相同 的 效果 ， 但 是 在 使 万 物 都 成 为 对 象 的 精神 中 ， 如 果 能 够 将 一 个 对 象 声 明 为 
是 “持久 性 ”的 ， 并 为 我 们 处 理 掉 所 有 细节 ， 那 将 会 显得 十 分 方便 。 

Java 的 对 象 序列 化 将 那些 实现 了 Serializable 接 口 的 对 象 转换 成 一 个 字 节 序列 ， 并 能 够 在 以 
后 将 这 个 字 节 序列 完全 恢复 为 原来 的 对 象 。 这 一 过 程 甚至 可 通过 网 络 进 行 ， 这 意味 着 序列 化 机 
制 能 自动 弥补 不 同 操作 系统 之 间 的 差异 。 也 就 是 说 ， 可 以 在 运行 Windows 系 统 的 计算 机 上 创建 
一 个 对 象 ， 将 其 序列 化 ， 通 过 网 络 将 它 发 送 给 一 台 运行 Unix 系 统 的 计算 机 ， 然 后 在 那里 准确 地 
重新 组 装 ， 而 却 不 必 担 心 数据 在 不 同 机 器 上 的 表示 会 不 同 ， 也 不 必 关 心 字 节 的 顺序 或 者 其 他 任 
何 细节 。 

就 其 本 身 来 说 ， 对 象 的 序列 化 是 非常 有 趣 的 ， 因 为 利用 它 可 以 实现 轻 量 级 持久 性 
(lightweight persistence)。“ 持 久 性 ”意味 着 一 个 对 象 的 生存 周期 并 不 取决 于 程序 是 否 正在 执行 ， 
它 可 以 生存 于 程序 的 调用 之 间 。 通 过 将 一 个 序列 化 对 象 写 和 磁盘， 然后 在 重新 调用 程序 时 恢复 
该 对 象 ， 就 能 够 实现 持久 性 的 效果 。 之 所 以 称 其 为 “ 轻 量 级 "， 是 因为 不 能 用 某 种 “persistent” 
(持久 ) 关键 字 来 简单 地 定义 一 个 对 象 , 并 让 系统 自动 维护 其 他 细节 问题 (尽管 将 来 有 可 能 实现 )。 
相反 ， 对 象 必 须 在 程序 中 显 式 地 序列 化 (serialize) 和 反 序 列 化 还 原 (deserialize) 。 如 果 需 要 一 
个 更 严格 的 持久 性 机 制 ， 可 以 考虑 像 Hibernate 之 类 的 工具 (参见 http://hibernate.sourceforge.net)。 
更 多 的 细节 可 参考 《Thinking in Enterprise Java》， 该 书 可 从 www.MindView.com 下 载 。 

对 象 序列 化 的 概念 加 入 到 语言 中 是 为 了 支持 两 种 主要 特性 。 一 是 Java 的 远程 方法 调用 
(Remote Method Invocation, RMI) ， 它 使 存活 于 其 他 计算 机 上 的 对 象 使 用 起 来 就 像 是 存活 于 本 机 
上 一 样 。 当 向 远程 对 象 发 送 消息 时 ， 需 要 通过 对 象 序列 化 来 传输 参数 和 返回 值 。 在 《Thinking in 
Enterprise Java》 中 有 对 RMI 的 具体 讨论 。 

再 者 ， 对 Java Beans 来 说 ， 对 象 的 序列 化 也 是 必需 的 〈 可 参看 第 14 章 )。 使 用 一 个 Bean 时 ， 
一 般 情 况 下 是 在 设计 阶段 对 它 的 状态 信息 进行 配置 。 这 种 状态 信息 必须 保存 下 来 ， 并 在 程序 启 
动 时 进行 后 期 恢复 ， 这 种 具体 工作 就 是 由 对 象 序列 化 完成 的 。 

只 要 对 象 实现 了 Serializable 接 口 〈 该 接口 仅 是 一 个 标记 接口 ， 不 包括 任何 方法 ) ， 对 象 的 序 
列 化 处 理 就 会 非常 简单 。 当 序列 化 的 概念 被 加 入 到 语言 中 时 ， 许 多 标准 库 类 都 发 生 了 改变 ， 以 
便 具 备 序列 化 特性 一 其 中 包括 所 有 基本 数据 类 型 的 封装 器 、 所 有 容器 类 以 及 许多 其 他 的 东西 。 
甚至 Class 对 象 也 可 以 被 序列 化 。 

要 序列 化 一 个 对 象 ， 首 先 要 创建 某 些 OutputStream 对 象 ， 然 后 将 其 封装 在 一 个 ObjectOutput- 
Stream 对 象 内 。 这 时 ， 只 需 调用 writeObject0 即 可 将 对 象 序列 化 ， 并 将 其 发 送 给 OutputStream 
(对 象 化 序列 是 基于 字 节 的 ， 因 要 使 用 InputStream 和 OutputStream 继 承 层次 结构 )。 要 反 向 进行 
该 过 程 ( 即 将 一 个 序列 还 原 为 一 个 对 象 )， 需 要 将 一 个 InputStream 封 装 在 ObjectInputStream 内 ， 
然后 调用 readObjeetO0。 和 往常 一 样 ， 我 们 最 后 获得 的 是 一 个 引用 ， 它 指向 一 个 向 上 转型 的 
Object， 所 以 必须 向 下 转型 才能 直接 设置 它们 。 

对 象 序列 化 特别 “聪明 ”的 一 个 地 方 是 它 不 仅 保存 了 对 象 的 “全 景 图 "， 而 且 能 追踪 对 象 内 
所 包含 的 所 有 引用 ， 并 保存 那些 对 象 ， 接 着 又 能 对 对 象 内 包含 的 每 个 这 样 的 引用 进行 追踪 ， 依 
此 类 推 。 这 种 情况 有 时 被 称 为 “对 象 网 ”， 单 个 对 象 可 与 之 建立 连接 ， 而 且 它 还 包含 了 对 象 的 引 
用 数组 以 及 成 员 对 象 。 如 果 必 须 保持 一 套 自己 的 对 象 序列 化 机 制 ， 那 么 维护 那些 可 追踪 到 所 有 
链接 的 代码 可 能 会 显得 非常 麻烦 。 然 而 ， 由 于 Java 的 对 象 序列 化 似乎 找 不 出 什么 缺点 ， 所 以 请 
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尽量 不 要 自己 动手 ， 让 它 用 优化 的 算法 自动 维护 整个 对 象 网 。 下 面 这 个 例子 通过 对 链接 的 对 象 
生成 一 个 worm (蠕虫 ) 对 序列 化 机 制 进行 了 测试 。 每 个 对 象 都 与 worm 中 的 下 一 段 链接 ， 同 时 又 
与 属于 不 同类 (Data) 的 对 象 引用 数组 链接 : 


//: to/Worm.java 
// Demonstrates object serialization. 
import java.i 
import jav: 
import stal 








tit.’ 
net .mindview.util.Print.*; 





class Data implements Serializable { 
private int n; 
public Data(int n) { this.n =n; } 
public String toString() { return Integer.toString(n); } 
} 


public class Worm implements Serializable { 
private static Random rand = new Random(47); 
private Data(] d = { 
new Data(rand.nextInt(10)), 
new Data(rand.nextInt(10)), 
new Data(rand.nextInt(10)) 
i 
private Worm next; 
private char c; 
// Value of i == number of segments 
public Worm(int i, char x) { 
print("Worm constructor: ”+ i); 
c= 
if( > 9) 
next = new Worm(i, (char) (x + 1)); 








} 
public Worm() { 
print("Default constructor"); 
} 
public String toString() { 
StringBuilder result = new StringBuilder (": 
result. append(c) ; 
result. append("("); 
for(Data dat : d) 
result. append(dat) ; 
result. append(")"); 
if(next != null) [982 
result. append(next) ; 
return result. toString(); 
} 
public static void main(String{] args) 
throws ClassNotFoundException, IOException { 
Worm w = new Worm(6, 'a'); 
print("w =" +w); 
ObjectOutputStream out = new ObjectOutputStream( 
new FileOutputStream("worm.out")); 
out.writeObject ("Worm storage\n"); 
out .writeObject (w); 
out.close(); // Also flushes output 
ObjectInputStream in = new ObjectInputStream( 
new FileInputStream("worm.out")); 
String s = (String)in. readObject(); 
Worm w2 = (Worm) in. readObject(); 
print(s + "W2 = " + w2); 
ByteArrayOutputStream bout 
new ByteArrayOutputStream( 
ObjectOutputStream out2 = new ObjectOutputStream(bout) : 
out2.writeObject("Worm storage\n™ 
out2.writeObject (w): 
out2. flush(); 









































983 
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ObjectInputStream in2 = new ObjectInputStream( 
new ByteArrayInputStream(bout . toByteArray())); 
s = (String) in2.readObject(): 
Worm w3 = (Worm) in2.readObject(); 
print(s + "w3 = " + w3); 
} 
) /* Output: 
Worm constructor: 
Worm constructor: 
Worm constructor: 
Worm constructor: 
Worm constructor: 
Worm constructor: 
w = :a(853) :b(119) :c (802) :d(788) :e (199) : f (881) 
Worm storage 
w2 = :a(853):b(119) :¢ (802) :d (788) :e(199) :f (881) 
Worm storage 
w3 = :a(853):b(119) :c (802) :d(788) :e(199) : f (881) 
“Ii~ 


更 有 趣 的 是 ，Worm 内 的 Data 对 象 数组 是 用 随机 数 初始 化 的 〈 这 样 就 不 用 怀疑 编译 器 保留 
了 某 种 原始 信息 )。 每 个 Worm 段 都 用 一 个 char 加 以 标记 。 该 char 是 在 递归 生成 链接 的 Worm 列 表 
时 自动 产生 的 。 要 创建 一 个 Worm, 必须 告诉 构造 器 你 所 希望 的 它 的 长 度 。 在 产生 下 一 个 引用 时 ， 
要 调用 Worm 构 造 器 ， 并 将 长 度 减 1， 以 此 类 推 。 最 后 一 个 next 引 用 则 为 null ( 空 )， 表 示 已 到 达 
Worm 的 尾部 。 

以 上 这 些 操作 都 使 得 事情 变 得 更 加 复杂 ， 从 而 加 大 了 对 象 序列 化 的 难度 。 然 而 ， 真 正 的 序 
列 化 过 程 却 是 非常 简单 的 。 一 旦 从 另外 某 个 流 创建 了 ObjectOutputStream，writeObject0 就 会 
将 对 象 序列 化 。 注 意 也 可 以 为 一 个 String 调 用 writeObject0。 也 可 以 用 与 DataOutputStream 相 同 
的 方法 写 人 所 有 基本 数据 类 型 (它们 具有 同样 的 接口 )。 

有 两 段 看 起 来 相似 的 独立 的 代码 。 一 个 读 写 的 是 文件 ， 而 另 一 个 读 写 的 是 字 节 数组 
(ByteArray)。 可 利用 序列 化 将 对 象 读 写 到 任何 DataInputStream 或 者 DataOutputStream， 甚 至 
包括 网 络 (正如 在 《Thinking in Enterprise Java》 中 所 述 ) 。 

从 输出 中 可 以 看 出 ， 被 还 原 后 的 对 象 确实 包含 了 原 对 象 中 的 所 有 链接 。 

注意 在 对 一 个 Serializable 对 象 进行 还 原 的 过 程 中 ， 没 有 调用 任何 构造 器 ， 包 括 默认 的 构造 
器 。 整 个 对 象 都 是 通过 从 InputStream 中 取得 数据 恢复 而 来 的 。 

练习 27，(1) 创建 一 个 Serializable 类 ， 它 包含 一 个 对 第 二 个 Serializable 类 的 对 象 的 引用 。 创 
建 你 的 类 的 实例 ， 将 其 序列 化 到 硬盘 上 ， 然 后 恢复 它 ， 并 验证 这 个 过 程 可 以 正确 地 工作 。 
18.12.1 导 找 类 

读者 或 许 会 奇怪 ， 将 一 个 对 象 从 它 的 序列 化 状态 中 局 复出 来 ， 有 哪些 工作 是 必须 的 昵 ? 举 
个 例子 来 说 ， 假 如 我 们 将 一 个 对 象 序列 化 ， 并 通过 网 络 将 其 作为 文件 传送 给 另 一 台 计算 机 ， 那 
么 ， 另 一 台 计 算 机 上 的 程序 可 以 只 利用 该 文件 内 容 来 还 原 这 个 对 象 吗 ? 

回答 这 个 问题 的 最 好 方法 就 是 做 一 个 实验 。 下 面 这 个 文件 位 于 本 章 的 子 目 录 下 : 


44: io/Alien.java 

// A serializable class. 

import java.io.*; 

public class Alien implements Serializable {) ///:~ 


而 用 于 创建 和 序列 化 一 个 Alien 对 象 的 文件 也 位 于 相同 的 目录 下 ; 


//: i0/FreezeAlien.java 
// Create a serialized output file. 
import java.io.*; 


Puwewa 
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public class FreezeAlien { 
public static void main(String[] args) throws Exception { 
ObjectOutput out = new ObjectOutputStream( 
new FileOutputStream("X.file")); 
Alien quellek = new Alien(); 
out wri teObject (quel lek) ; 


} 
dU 
这 个 程序 不 但 能 捕获 和 处 理 异常 ， 而 且 将 异常 抛 出 到 main0) 方 法 之 外 ， 以 便 通过 控制 台 产 
生 报 告 。 一 旦 该 程序 被 编译 和 运行 ， 它 就 会 在 c12 目 录 下 产生 一 个 名 为 X.file 的 文件 。 以 下 代码 位 
于 一 个 名 为 xfiles 的 子 目录 下 : 


//: 40/xfiles/ThawAlien. java 

// Try to recover a serialized file without the 
// class of object that's stored in that file. 
/1/ {RunByHand} 

import java.io.*; 


public class ThawAlien { 
public static void main(String(] args) throws Exception { 
ObjectInputStream in = new Object InputStream( 
new FileInputStream(new File("..", "X.file"))); 
Object mystery = in.readObject(): 
System. out.println(mystery.getClass()) ; 





} ;» Output: 

class Alien 

“i~ 

打开 文件 和 读 取 mystery 对 象 中 的 内 容 都 需要 Alien 的 Class 对 象 ， 而 Java 虚 拟 机 找 不 到 
Alien.class (除非 它 正好 在 类 路 径 Classpath 内 ， 而 本 例 却 不 在 类 路 径 之 内 ) 。 这 样 就 会 得 到 一 个 
名 叫 ClassNotFoundException 的 异常 同样， 除非 能 够 验证 Alien 存 在 ， 否 则 它 等 于 消失 )。 必 须 
保证 Java 虚 拟 机 能 找到 相关 的 ,class 文件 。 


18.12.2 序列 化 的 控制 

正如 大 家 所 看 到 的 ， 默 认 的 序列 化 机 制 并 不 难 操纵 。 然 而 ， 如 果 有 特殊 的 需要 那 又 该 怎么 
办 呢 ? 例如 ， 也 许 要 考虑 特殊 的 安全 问题 ， 而 且 你 不 希望 对 象 的 某 一 部 分 被 序列 化 ， 或 者 一 个 
对 象 被 还 原 以 后 ， 某 子 对 象 需要 重新 创建 ， 从 而 不 必 将 该 子 对 象 序列 化 。 

在 这 些 特殊 情况 下 ， 可 通过 实现 Externalizable 接 口 一 一 代 赫 实现 Serializable 接 口 一 来 对 
序列 化 过 程 进行 控制 。 这 个 Externalizable 接 口 继承 了 Serializable 接 口 ， 同 时 增添 了 两 个 方法 : 
writeExternal0 和 readExternal0)。 这 两 个 方法 会 在 序列 化 和 反 序列 化 还 原 的 过 程 中 被 自动 调用 ， 
以 便 执行 一 些 特殊 操作 。 

下 面 这 个 例子 展示 了 Externalizable 接 口 方法 的 简单 实现 。 注 意 Blipl 和 Blip2 除 了 细微 的 差 
别 之 外 ， 几 乎 完全 一 致 (研究 一 下 代码 ， 看 看 你 能 否 发 现 ) : 


//: io/BLips.java 

// Simple use of Externalizable & a pitfall. w 
import java.io.*; 

import static net.mindview.util.Print.*; 


class Blip1 implements Externalizable { 
public Blip1() { 
print("BLipl Constructor"); 


} 
public void writeExternal(ObjectOutput out) 
throws IOException { 
print("Blipl.writeéxternal"); 
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} 
public void readExternal(ObjectInput in) 
throws IOException, ClassNotFoundException { 
print("Blipl.readExternal") ; 
} 








986] } 
class Blip2 implements Externalizable { 
Blip20 { 
print("Blip2 Constructor"); 








public void writeExternal (ObjectOutput out) 
throws IOException { 
print("Blip2.writeExternal”) ; 


public void readExternal(ObjectInput in) 
throws IOException, ClassNotFoundException { 
print ("B1ip2.readExternal"); 


} 


public class Blips { 
public static void main(String(] args) 
throws IOException, ClassNotFoundexception { 
print("Constructing objects:"); 
Blipl bl = new Blip1(); 
Blip2 b2 = new Blip2(); 
ObjectOutputStream o = new ObjectOutputStream( 
new FileOutputStream("Blips.out")); 
print("Saving objects:"); 
o.writedbject (b1) ; 
o.writedbject (b2) ; 
o.close(); 
// Now get them back: 
ObjectInputStream in = new ObjectInputStream( 
new FileInputStream("B1ips.out")); 
print("Recovering b1:"); 
bl = (Blip) in. readObject(); 
// OOPS! Throws an exception: 
//! print("Recovering b2 
/1/1! b2 = (Blip2)in, readObject(); 
} 





} /* Output: 
Constructing objects: 
Blipl Constructor 
Blip2 Constructor 
Saving objects: 
Blipl:writeExternal 
Blip2.writeExternal 
Recovering b1: 

Blipl Constructor 


Blip1.read€xternal 
Whim 


上 例 中 没有 恢复 Blip2 对 象 ， 因 为 那样 做 会 导致 一 个 异常 。 你 找 出 Blip1 和 Blip2 之 间 的 区 
别 了 吗 ? Blip1 的 构造 器 是 “公共 的 ”(public) ，Blip2 的 构造 器 却 不 是 ， 这 样 就 会 在 恢复 时 造 
成 异常 。 试 试 将 Blip2 的 构造 器 变 成 public 的 ， 然 后 删除 //! 注 释 标记 ， 看 看 是 否 能 得 到 正确 的 
结果 。 

恢复 bl 后 ， 会 调用 Blip1 默 认 构造 器 。 这 与 恢复 一 个 Serializable 对 象 不 同 。 对 于 Serializable 对 
象 ， 对 象 完全 以 它 存储 的 二 进 制 位 为 基础 来 构造 ， 而 不 调用 构造 器 。 而 对 于 一 个 Externalizable 对 
象 ， 所 有 普通 的 默认 构造 器 都 会 被 调用 (包括 在 字段 定 义 时 的 初始 化 )， 然后 调用 readExternal0。 
必须 注意 这 一 点 一 一 所 有 默认 的 构造 器 都 会 被 调用 ， 才 能 使 Externalizable 对 象 产 生 正确 的 行为 。 

下 面 这 个 例子 示范 了 如 何 完整 保存 和 恢复 一 个 Externalizable 对 象 : 
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11; io/Blip3. java 
// Reconstructing an externalizable object. 
import java.io.*; 


import static net.mindview.util.Print.*; 





public class Blip3 implements Externalizable { 
private int i; 
private String s; // No initialization 
public Blip3() { 
print("Blip3 Constructor"); 
// s, 1 not initialized 
} 
public Blip3(String x, int a) { 
print("Blip3(String x, int a)"); 
Ss =x; 
1=a; 
1/ s & i initialized only in non-default constructor. 
} 
public String toString() { return s + i; } 
public void writeExternal(ObjectOutput out) 
throws IOException { 
print("Blip3.writeExternal"); 
// You must do this: 
out .writedbject(s); 
out .writeInt (i); 
} 
public void readExternal (ObjectInput in) 
throws IOException, ClassNotFoundException { 
print("Blip3.readExternal"); 
// You must do this: 
s = (String)in. readObject(); 
i = in.readInt(); 
} 


public static void main(String(] args) 
throws IOException, ClassNotFoundException { 
print ("Constructing objects:"); 
Blip3 b3 = new Blip3("A String ", 47); 
print (b3); 
ObjectOutputStream o = new ObjectOutputStream( 
new FileOutputStream("B1ip3.out")); 
print("Saving object:"); 
0.writeObject (b3) ; 
0.close(); 
// Now get it back: 
ObjectInputStream in = new ObjectInputStream( 
new FileInputStream("Blip3.out”)); 
print("Recovering b3:"); 
b3 = (Blip3) in. readObject(); 
print (b3); 
} 
} /* Output: 
Constructing objects: 
Blip3(String x, int a) 
A String 47 
Saving object: 
Blip3.writeExternal 
Recovering b3: 
Blip3 Constructor 
Blip3.readExternal 
A String 47 
Wh 


其 中 ， 字 段 s 和 i 只 在 第 二 个 构造 器 中 初始 化 ， 而 不 是 在 默认 的 构造 器 中 初始 化 。 这 意味 着 假 
如 不 在 readExternal0 中 初始 化 和 i，s 就 会 为 null， 而 就 会 为 零 ( 因 为 在 创建 对 象 的 第 一 步 中 将 
对 象 的 存储 空间 清理 为 0) 。 如 果 注 释 掉 跟随 于 “You must do this” 后 面 的 两 行 代码 ， 然 后 运行 
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我 们 如 果 从 一 个 Externalizable 对 象 继承 ， 通 常 需要 调用 基 类 版 本 的 writeExternal0 和 read- 
External0 来 为 基 类 组 件 提供 恰当 的 存储 和 恢复 功能 。 

因此 ， 为 了 正常 运行 ， 我 们 不 仅 需 要 在 writeExternal() 方 法 (没有 任何 默认 行为 来 为 
Externalizable 对 象 写 入 任何 成 员 对 象 ) 中 将 来 自 对 象 的 重要 信息 写 入 ， 还 必须 在 readExternal0 
方法 中 恢复 数据 。 起 先 ， 可 能 会 有 一 点 迷惑 ， 因 为 Externalizable 对 象 的 默认 构造 行为 使 其 看 起 
来 似乎 像 某 种 自动 发 生 的 存储 与 恢复 操作 。 但 实际 上 并 非 如 此 。 

练习 28: (2) 复制 Blips.java 并 重 命名 为 BlipCheck.java， 然 后 将 类 Blip2 重 命名 为 BlipCheck 
(使 其 成 为 pablic 的 ， 并 在 此 过 程 中 删除 类 Blips 中 的 公共 作用 域 )。 删 除 文件 中 的 /标记 ， 然 后 执 
行 含有 这 几 个 错误 行 的 程序 。 接 下 来 ， 注 释 掉 BlipCheck 的 默认 构造 器 。 执 行 之 并 解释 它 可 以 运 
行 的 原因 。 注 意 编译 后 我 们 必须 使 用 java Blips 执 行程 序 ， 因 为 main0 方 法 仍 在 类 Blips 中 。 

练习 29: (2) 注释 掉 Blip3.java 中 自 “You must do this:” 开 始 的 两 行 ， 运 行 之 。 解 释 结果 ， 
并 说 出 该 结果 与 这 两 行 在 程序 中 运行 时 所 产生 的 结果 不 同 的 原因 。 

transient (Rt) 关键 字 

当 我 们 对 序列 化 进行 控制 时 ， 可 能 某 个 特定 子 对 象 不 想 让 Java 的 序列 化 机 制 自动 保存 与 恢 
复 。 如 果子 对 象 表示 的 是 我 们 不 希望 将 其 序列 化 的 敏感 信息 (如 密码 ) ， 通 常 就 会 面临 这 种 情况 。 
即使 对 象 中 的 这 些 信息 是 private (私有 ) 属性 ， 一 经 序列 化 处 理 ， 人 们 就 可 以 通过 读 取 文件 或 
者 拦截 网 络 传输 的 方式 来 访问 到 它 。 

有 一 种 办 法 可 防止 对 象 的 敏感 部 分 被 序列 化 ， 就 是 将 类 实现 为 Externalizable， 如 前 面 所 示 。 
这 样 一 来 ， 没 有 任何 东西 可 以 自动 序列 化 ， 并 且 可 以 在 writeExternal0 内 部 只 对 所 需 部 分 进行 显 
式 的 序列 化 。 

然而 ， 如 果 我 们 正在 操作 的 是 一 个 Serializable 对 象 ， 那 么 所 有 序列 化 操作 都 会 自动 进行 。 
为 了 能 够 予以 控制 ， 可 以 用 transient (HH) 关键 字 逐 个 字段 地 关闭 序列 化 ， 它 的 意思 是 “不 
用 麻烦 你 保存 或 恢复 数据 一 我 自己 会 处 理 的 "。 

例如 ， 假 设 某 个 Login 对 象 保存 某 个 特定 的 登录 会 话 信息 。 登 录 的 合法 性 通过 校 验 之 后 ， 我 
们 想 把 数据 保存 下 来 ， 但 不 包括 密码 。 为 做 到 这 一 点 ， 最 简单 的 办 法 是 实现 Serializable， 并 将 
password 字 段 标志 为 transient。 下 面 是 具体 的 代码 ; 


11: io/Logon.java 
// Demonstrates the "transient" keyword. 
import java.util.concurrent.*; 

import java. i 
import java.u i 
import static net.mindview.util.Print.*: 





public class Logon implements Serializable ( 
private Date date = new Date(); 
private String username; 
private transient String password; 
public Logon(String name, String pwd) { 
username = name; 
Password = pwd; 


} 
public String toString() { 
return “logon info: \n username: ”+ username + 
"\n date: "+ date + "\n password: "+ password; 
} 


Public static void main(String{] args) throws Exception { 
Logon a = new Logon("Hulk", "myLittlePony") : 
print("logon a = " + a); 
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ObjectOutputStream o = new ObjectOutputStream( 
new FileQutputStream("Logon.out")); 

o.writeObject (a); 

o.close(); 

TimeUnit. SECONDS. sieep(1); // Delay 

// Now get them back: 

ObjectInputStream in = new ObjectInputStream( 
new FileInputStream("Logon.out")); 

print("Recovering object at " + new Date()); 

a = (Logon)in.readObject(): 

print("logon a = " + a); 


} 
} /* Output: (Sample) 
logon a = logon info: 
username: Hulk 
date: Sat Nov 19 15:03:26 MST 2005 
password: myLittlePony 
“Recovering object at Sat Nov 19 15:03:28 MST 2005 
logon a = logon info: 
username: Hulk 
date: Sat Nov 19 15:63:26 MST 2005 
password: null 
Wh 


可 以 看 到 ， 其 中 的 date 和 username 域 是 一 般 的 〈 不 是 transient 的 ) ， 所 以 它们 会 被 自动 序列 
化 。 而 password 是 transient 的 ， 所 以 不 会 被 自动 保存 到 磁盘 ， 另 外 ， 自 动 序列 化 机 制 也 不 会 党 
试 去 恢复 它 。 当 对 象 被 恢复 时 ，password 域 就 会 变 成 null。 注 意 ， 虽 然 toString0 是 用 ， 重 载 后 
的 + 运算 符 来 连接 String 对 象 ， 但 是 null 引 用 会 被 自动 转换 成 字符 串 null。 

我 们 还 可 以 发 现 : date 字 段 被 存储 了 到 磁盘 并 从 磁盘 上 被 恢复 了 出 来 ， 而 且 没有 再 重新 生成 。 

由 于 Externalizable 对 象 在 默认 情况 下 不 保存 它们 的 任何 字段 ， 所 以 transient 关 键 字 只 能 和 
Serializable 对 象 一 起 使 用 。 

Externalizable 的 蔡 代 方法 

如 果 不 是 特别 坚持 实现 Externalizable 接 口 ， 那 么 还 有 另 一 种 方法 。 我 们 可 以 实现 Serializa- 
ble 接 口 ， 并 添加 (注意 我 说 的 是 “添加 ”， 而 非 “覆盖 ”或 者 “实现 ”") 名 为 writeObjectO 
和 readObject0 的 方法 。 这 样 一 旦 对 象 被 序列 化 或 者 被 反 序列 化 还 原 ， 就 会 自动 地 分 别 调用 
这 两 个 方法 。 也 就 是 说 ， 只 要 我 们 提供 了 这 两 个 方法 ， 就 会 使 用 它们 而 不 是 默认 的 序列 化 
机 制 。 

这 些 方法 必须 具有 准确 的 方法 特征 签名 : 


private void write0bject(ObjectOutputStream stream) 
throws IOException; 


private void readObject (ObjectInputStream stream) 
throws IOException, ClassNotFoundException 


从 设计 的 观点 来 看 ， 现 在 事情 变 得 真是 不 可 思议 。 首 先 ， 我 们 可 能 会 认为 由 于 这 些 方法 不 是 
基 类 或 者 Serializable 接 口 的 一 部 分 ， 所 以 应 该 在 它们 自己 的 接口 中 进行 定义 。 但 是 注意 它们 被 定 
义 成 了 private， 这 意味 着 它们 仅 能 被 这 个 类 的 其 他 成 员 调 用 。 然 而 ， 实 际 上 我 们 并 没有 从 这 个 类 
的 其 他 方法 中 调用 它们 ， 而 是 ObjectOutputStream 和 ObjectInputStream 对 象 的 writeObject0 和 
readObject0 方 法 调用 你 的 对 象 的 writeObject0 和 readObject0 方 法 注意 关于 这 里 用 到 的 相同 方 
法 名 ， 我 尽量 抑制 住 不 去 说 骂 。 一 名 话 : 混乱 ) 。 读 者 可 能 想 知道 ObjectOutputStream 和 
ObjectInputStream 对 象 是 怎样 访问 你 的 类 中 的 private 方 法 的 。 我 们 只 能 假设 这 正 是 序列 化 神奇 
的 一 部 分 e 。 


日 149 节 展示 了 如 何在 类 的 外 部 访问 private 方 法 。 
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在 接口 中 定义 的 所 有 东西 都 自动 是 public 的 ， 因 此 如 果 writeObject0 和 readObject0 必 须 是 
Private 的， 那么 它们 不 会 是 接口 的 一 部 分 。 因 为 我 们 必须 要 完全 遵循 其 方法 特征 签名 ， 所 以 其 
效果 就 和 实现 了 接口 一 样 。 

在 调用 ObjectOutputStream.writeObject0 时 ， 会 检查 所 传递 的 Serializable 对 象 ， 看 看 是 否 
实现 了 它 自己 的 writeObject0。 如 果 是 这 样 , 就 跳 过 正常 的 序列 化 过 程 并 调用 它 的 writeObject0。 
readObject0 的 情形 与 此 相同 。 

还 有 另外 一 个 技巧 。 在 你 的 writeObjeet0 内 部 ， 可 以 调用 defaultWriteObjeet0 来 选择 执行 
默认 的 writeObject0。 类 似 地 ， 在 readObject0 内 部 ， 我 们 可 以 调用 defaultReadObject0。 下 面 
这 个 简单 的 例子 演示 了 如 何 对 一 个 Serializable 对 象 的 存储 与 恢复 进行 控制 : 


/1/: io/SerialCtl.java 
// Controlling serialization by adding your own 
// writeObject() and readObject() methods. 
import java.io.*; 


public class SerialCtl implements Serializable { 
private String a; 
private transient String b; 
public SerialCtl(String aa, String bb) { 
a = "Not Transient: ”+ aa; 
b = "Transient: ”+ bb; 





) 

public String toString() { return a + "\n" + b; } 

private void writeObject (ObjectOutputStream stream) 

throws IOException { 
stream.defaultwriteObject(); 
stream.writedbject (b) ; 

} 

private void readObject(ObjectInputStream stream) 

throws IOException, ClassNotFoundException { 
stream.defaultReadObject() ; 
b = (String)stream. readObject () ; 

} 

public static void main(String{] args) 

throws IOException, ClassNotFoundException { 
SerialCtl sc = new SerialCtl("Test1", "Test2"); 
System.out.printin("Before:\n" + sc): 
ByteArrayOutputStream buf= new ByteArrayOutputStream(); 
ObjectOutputStream o = new ObjectOutputStream(buf) ; 
o.writeObject (sc) ; 
// Now get it back: 
ObjectInputStream in = new ObjectInputStream( 

new ByteArrayInputStream(buf.toByteArray())); 

SerialCtl sc2 = (SerialCtl)in.readObject(); 
System.out.printin("After:\n" + sc2); 


} 
} /* Output: 
Before: 
Not Transient: Testi 
Transient: Test2 
After: 
Not Transient: Test1 
Transient: Test2 
hx 


在 这 个 例子 中 ， 有 一 个 String 字 段 是 普通 字段 ， 而 另 一 个 是 transient 字 段 ， 用 来 证 明 非 
transient 字 段 由 defaultWriteObject0 方 法 保存 ， 而 transient 字 段 必 须 在 程序 中 明确 保存 和 恢复 。 
字段 是 在 构造 器 内 部 而 不 是 在 定义 处 进行 初始 化 的 ， 以 此 可 以 证 实 它 们 在 反 序列 化 还 原 期 间 没 

994] 有 被 一 些 自动 化 机 制 初始 化 。 
如 果 我 们 打算 使 用 默认 机 制 写 人 对 象 的 在 transient 部 分 ， 那 么 必须 调用 defaultWriteObject0 
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作为 writeObject0 中 的 第 一 个 操作 ， 并 让 defaultReadObject0 作 为 readObject0 中 的 第 一 个 操作 。 
这 些 都 是 奇怪 的 方法 调用 。 例 如 ， 如 果 我 们 正在 为 ObjectOutputStream 调 用 defaultWrite- 
Object0 且 没有 传递 任何 参数 ， 然 而 不 知 何故 它 却 可 以 运行 ， 并 且 知 道 对 象 的 引用 以 及 如 何 写 入 
非 transient 部 分 。 真 是 奇怪 之 极 。 

对 transient 对 象 的 存储 和 恢复 使 用 了 我 们 比较 熟悉 的 代码 。 请 再 考虑 一 下 在 这 里 所 发 生 的 
事情 。 在 main0 中 ， 创 建 SerialCt 对 象 ， 然 后 将 其 序列 化 到 ObjectOutputStream (注意 在 这 种 
情况 下 ， 使 用 的 是 缓冲 区 而 不 是 文件 一 这 对 于 ObjectOutputStream 来 说 是 完全 一 样 的 ) 。 序 列 
化 发 生 在 下 面 这 行 代码 当中 : 

oO.writeObject(sc); 

writeObject() 方 法 必须 检查 sc， 判断 它 是 否 拥有 自己 的 writeObjeet() 方 法 (不 是 检查 接 
口 一 这 里 根本 就 没有 接口 ， 也 不 是 检查 类 的 类 型 ， 而 是 利用 反射 来 真正 地 搜索 方法 )。 如 果 有 ， 
那么 就 会 使 用 它 。 对 readObject0 也 采用 了 类 似 的 方法 。 或 许 这 是 解决 这 个 问题 的 唯一 切实 可 行 
的 方法 ， 但 它 确实 有 点 古怪 。 

版 本 控制 

有 了 时 可 能 想 要 改变 可 序列 化 类 的 版 本 (比如 源 类 的 对 象 可 能 保存 在 数据 库 中 )。 虽 然 Java 支 
持 这 种 做 法 ， 但 是 你 可 能 只 在 特殊 的 情况 下 才 这 样 做 ， 此 外 ， 还 需要 对 它 有 相当 深 程度 的 了 解 
(在 这 里 我 们 就 不 再 试图 达到 这 一 点 )。 从 http:/java.sun.com 处 下 载 的 JDK 文 档 中 对 这 一 主题 进行 
了 非常 彻底 的 论述 。 

我 们 会 发 现在 JDK 文 档 中 有 许多 注解 是 从 下 面 的 文字 开始 的 : 


敬告 ”该 类 的 序列 化 对 象 和 未 来 的 Swing 版 本 不 兼容 。 当 前 对 序列 化 的 支持 只 适用 于 短 
期 存储 或 应 用 之 间 的 RMI。 


这 是 因为 Java 的 版 本 控制 机 制 过 于 简单 ， 不 能 在 任何 场合 都 可 靠 运转 ， 尤 其 是 对 
JavaBeans 更 是 如 此 。 有 关 人 员 正 在 设法 修正 这 一 设计 ， 也 就 是 警告 中 的 相关 部 分 。 
18.12.3 使 用 “持久 性 ” 

一 个 比较 诱 人 的 使 用 序列 化 技术 的 想法 是 ， 存 储 程序 的 一 些 状态 ， 以 便 我 们 随后 可 以 很 容 
易 地 将 程序 恢复 到 当前 状态 。 但 是 在 我 们 能 够 这 样 做 之 前 ， 必 须 回答 几 个 问题 。 如 果 我 们 将 两 
个 对 象 一 它们 都 具有 指向 第 三 个 对 象 的 引用 一 进行 序列 化 ， 会 发 生 什么 情况 ? 当 我 们 从 它们 
的 序列 化 状态 恢复 这 两 个 对 象 时 ， 第 三 个 对 象 会 只 出 现 一 次 吗 ? 如 果 将 这 两 个 对 象 序列 化 成 独 
立 的 文件 ， 然 后 在 代码 的 不 同 部 分 对 它们 进行 反 序列 化 还 原 ， 又 会 怎样 呢 ? 

下 面 这 个 例子 说 明了 上 述 问题 : 

//: io/ywortd.java 

import java.io.*; 


import java.utit.*; 
import static net.mindview.util.Print.*; 





class House implements Serializable {} 


class Animal implements Serializable { 
private String name; 
private House preferredHouse: 
Animal (String nm, House h) { 
name = nm; 
preferredHouse = h; 


} 
Public String toString() { 
return name + "[" + super.toString() + 
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"], " + preferredHouse + *\n"; 


public class MyWorld { 
public static void main(String[] args) 
throws IOException, ClassNotFoundException { 
House house = new House(); 
List<Animal> animals = new ArrayList<Animal>(); 
animals.add(new Animal("Bosco the dog", house)); 
animals.add(new Animal("Ralph the hamster", house)); 
animals.add(new Animal ("Molly the cat", house)); 
print("animals: ”+ animals); 
ByteArrayOutputStream bufl = 
new ByteArrayOutputStream() ; 
ObjectOutputStream ol = new ObjectOutputStream(buf 1) ; 
ol.writeObject (animals) ; 
ol.writeObject(animals); // Write a 2nd set 
// Write to a different stream: 
ByteArrayOutputStream buf2 = 
new ByteArrayOutputStream() ; 
ObjectOutputStream 02 = new ObjectOutputStream(buf2) ; 
02.writeObject (animals) ; 
// Now get them back: 
ObjectInputStream inl = new ObjectInputStream( 
new ByteArrayInputStream(bufl. toByteArray())); 
ObjectInputStream in2 = new ObjectInputStream( 
new ByteArrayInputStream(buf2. toByteArray())); 
List 
animalsl = (List)inl.readObject(), 
animals2 = (List)inl.readObject(), 
animals3 = (List)in2.readObject(); 
print("animatsi: ”+ animals); 
print("animals2: ”+ animals2); 
print("animals3: ”+ animals3); 





} 
} /* Output: (Sample) 
animals: [Bosco the doglAnimal@addbf1], House@42e816 
, Ralph the hamster [Anima1@9304b1], House@42e816 
, Molly the cat[Animal@190d11], House@42e816 
J 
animalsi: [Bosco the dog{Animal@de6f34], House@1S6ee8e 
, Ralph the hamster [Animal@47b486) , House@156ee8e 
, Molly the cat[Animal@19b49e6] , House@iSéee8e 
] 


animals2: [Bosco the dog{Animal@de6f34}, House@156ee8e 
. Ralph the hamster (Animal@47b480] , House@156ee8e 
Molly the cat{Animal@19b49e6] , House@iS6ee8e 


] 

animals3: [Bosco the dog[Animate16d448] ，Houseeegelc6 
, Ralph the hamster [Animal@6calc], House@edeicé 

, Molly the cat[Animal@1bf216a), House@eGeic6 


1 

Iiia 

这 里 有 一 件 有 趣 的 事 : 我 们 可 以 通过 一 个 字 节 数组 来 使 用 对 象 序列 化 ， 从 而 实现 对 任何 可 
Serializable 对 象 的 “深度 复制 ”(deep copy) 一 一 深度 复制 意味 着 我 们 复制 的 是 整个 对 象 网 ， 而 
不 仅仅 是 基本 对 象 及 其 引用 。 复 制 对 象 将 在 本 书 的 在 线 补充 材料 中 进行 深入 地 探讨 。 

在 这 个 例子 中 ，Animal 对 象 包含 有 House 类 型 的 字段 。 在 main() 方 法 中 ， 创 建 了 一 个 
Animal 列 表 并 将 其 两 次 序列 化 ， 分 别 送 至 不 同 的 流 。 当 其 被 反 序列 化 还 原 并 被 打印 时 ， 我 们 可 
以 看 到 所 示 的 执行 某 次 运行 后 的 结果 (每 次 运行 时 ， 对 象 将 会 处 在 不 同 的 内 存 地 址 )。 

当然 ， 我 们 期 望 这 些 反 序列 化 还 原 后 的 对 象 地 址 与 原来 的 地 址 不 同 。 但 请 注意 ， 在 
animals] 和 animals2 中 却 出 现 了 相同 的 地 址 ， 包 括 二 者 共享 的 那个 指向 House 对 象 的 引用 。 另 
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一 方面 ， 当 恢复 animals3 时 ， 系 统 无 法 知道 另 一 个 流 内 的 对 象 是 第 一 个 流 内 的 对 象 的 别名 ， 因 
此 它 会 产生 出 完全 不 同 的 对 象 网 。 

只 要 将 任何 对 象 序列 化 到 单一 流 中 ， 就 可 以 恢复 出 与 我 们 写 出 时 一 样 的 对 象 网 ， 并 且 没有 
任何 意外 重复 复制 出 的 对 象 。 当 然 ， 我 们 可 以 在 写 出 第 一 个 对 象 和 写 出 最 后 一 个 对 象 期 间 改变 
这 些 对 象 的 状态 ， 但 是 这 是 我 们 自己 的 事 ， 无 论 对 象 在 被 序列 化 时 处 于 什么 状态 (无论 它们 和 
其 他 对 象 有 什么 样 的 连接 关系 )， 它 们 都 可 以 被 写 出 。 

如 果 我 们 想 保存 系统 状态 ， 最 安全 的 做 法 是 将 其 作为 “原子 ”操作 进行 序列 化 。 如 果 我 们 
序列 化 了 某 些 东西 ， 再 去 做 其 他 一 些 工 作 ， 再 来 序列 化 更 多 的 东西 ， 如 此 等 等 ， 那 么 将 无 法 安 
全 地 保存 系统 状态 。 取 而 代 之 的 是 ， 将 构成 系统 状态 的 所 有 对 象 都 置 入 单一 容器 内 ， 并 在 一 个 
操作 中 将 该 容器 直接 写 出 。 然 后 同样 只 需 一 次 方法 调用 ， 即 可 以 将 其 恢复 。 

下 面 这 个 例子 是 一 个 想象 的 计算 机 辅助 设计 (CAD) 系统 ， 该 例 演示 了 这 一 方法 。 此 外 ， 
它 还 引入 了 static 字 段 的 问题 ， 如 果 我 们 查看 JDK 文 档 ， 就 会 发 现 Class 是 Serializable 的 ， 因 此 只 
需 直 接 对 Class 对 象 序列 化 ， 就 可 以 很 容易 地 保存 static 字 段 。 在 任何 情况 下 ， 这 都 是 一 种 明智 的 
做 法 。 


//: \o/StoreCADState. java 
// Saving the state of a pretend CAD system. 
‘import java.io.*: 

import java.util.*; 


abstract class Shape implements Serializable { 
public static final int RED = 1, BLUE = 2, GREEN = 3; 
private int xPos, yPos, dimension; 
private static Random rand = new Random(47) ; 
private static int counter = 9; 
public abstract void setColor(int newColor); 
public abstract int getColor(); 
public Shape(int xVal, int yVal, int dim) { 
xPos = xVal; 
yPos = yVal; 
dimension = dim; 


} 
public String toString() { 
return getClass() + 
"color[" + getColor() + "] xPos[" + xPos + 
“) yPos[{" + yPos + "】 dimi" + dimension + "}\n"; 
} 
public static Shape randomFactory() { 
int xVal = rand.nextInt (108) ; 
int yVal = rand.nextInt (168) ; 
int dim = rand.nextInt (100); 
switch(counter++ % 3) { 
default: 
case 9: return new Circle(xVal, yVal, dim); 
case 1: return new Square(xVal, yVal, dim); 
case 2: return new Line(xVal, yal, dim); 
} 
} 
} 


class Circle extends Shape { 
private static int color = RED; 
public Circle(int xVal, int yVal, int dim) { 
super(xVal, yVal, dim); 


} 
public void setColor(int newCotor) { color = newColor; } 
public int getColor() { return color; } 














1999] 
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class Square extends Shape { 
private static int color; 
public Square(int xVal, int yVal, int dim) { 
super (xVal, yVal, dim); 
color = RED: 
y 
public void setColor(int newColor) { color = newColor; } 
public int getColor() { return color; } 
} 


class Line extends Shape { 
private static int color = RED; 
public static void 
serializeStaticState(ObjectOutputStream os) 
throws IOException { os.writeInt(color); } 
public static void 
deserializeStaticState(ObjectInputStream os) 
throws IOException { color = os.readInt(); } 
public Line(int xVal, int yVal, int dim) { 

super(xVal, yVal, dim); 


public void setColor(int newColor) { color = newColor; } 
public int getColor() { return color; } 
} 


public class StoreCADState { 
public static void main(String{] args) throws Exception { 
List<Class<? extends Shape>> shapeTypes = 
new ArrayList<Class<? extends Shape>>(); 
// Add references to the class objects: 
shapeTypes .add(Circle.class) ; 
shapeTypes . add(Square.class) ; 
shapeTypes .add(Line.class) ; 
List<Shape> shapes = new ArrayList<Shape>(); 
// Make some shapes: 
for(int i = 0; 1 < 10; 1++) 
Shapes .add(Shape.randomFactory()); 
// Set all the static colors to GREEN: 
for(int i = 0; 1 < 10; i++) 
((Shape) shapes. get (i)) .setColor (Shape. GREEN) ; 
// Save the state vector: 
ObjectOutputStream out = new ObjectOutputStream( 
new FileOutputStream("CADState.out")); 
out .writeObject (shapeTypes) ; 
Line. serializeStaticState (out); 
out .writeObject (shapes) ; 
// Display the shapes: 
System.out .printin(shapes) ; 
} 
} /* Output: 
[class Circtecotor[3] xPos{58] yPos[55] dim[93] 
, class Squarecolor{3] xPos(61] yPos[61] dim[29] 
, class Linecolor[3] xPos[68] yposI6] dim(22] 
, Class Circlecolor [3] xPos{7] yPos{88] dim[28] 
. class Squarecolor[3] xPos[51] yPos(89] dim[9] 
, Class Linecotor[3] xPos{78] ypos[98] dim[61] 
. class Circlecolor{3] xPos[20] yPos(58] dim[16] 
+ Class Squarecolor{3] xPos[40] yPos{11] dim[22] 
, class Linecolor{3] xPos{4] yPos[83] dim[6] 
, class Circlecolor[3] xPos[75] yPos[10] dim142] 
1 
Wha 


Shape 类 实现 了 Serializable， 所 以 任何 自 Shape 继 承 的 类 也 都 会 自动 是 Serializable 的 。 每 个 





Shape 都 含有 数据 ， 而 且 每 个 派生 自 Shape 的 类 都 包含 一 个 static 字 段 ， 用 来 确定 各 种 Shape 类 型 
的 颜色 (如 果 将 static 字 段 置信 基 类 ， 只 会 产生 一 个 static 字 段 ， 因 为 static 字 段 不 能 在 派生 类 中 
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复制 )。 可 对 基 类 中 的 方法 进行 重 载 ， 以 便 为 不 同 的 类 型 设置 颜色 (static 方 法 不 会 动态 绑 定 ， 
所 以 这 些 都 是 普通 的 方法 )。 每 次 调用 randomFactory0 方 法 时 ， 它 都 会 使 用 不 同 的 随机 数 作为 
Shape 的 数据 ， 从 而 创建 不 同 的 Shape。 
在 main0 中 ， 一 个 ArrayList 用 于 保存 Class 对 象 ， 而 另 一 个 用 于 保存 几何 形状 。 
恢复 对 象 相当 直观 ; 


//: io/RecoverCADState. java 

// Restoring the state of the pretend CAD system. 
// {RunFirst: StoreCADState} 

import java.io.*; 
import java.util.*; 





public class RecoverCADState { 
@SuppressWarnings ("unchecked") 
public static void main(String[] args) throws Exception { 
ObjectInputStream in = new ObjectInputStream( 
new FileInputStream("CADState.out")); 
// Read in the same order they were written: 
List<Class<? extends Shape>> shapeTypes = 
(List<Class<? extends Shape>>) in. readObject(); 
Line. deserializeStaticState(in) ; 
List<Shape> shapes = (List<Shape>) in. readObject(): 
System.out.printin(shapes) ; 
} 
) /* Output: 
(class Circlecolor(1] xPos[58] ypos[55] dim[93] 
, Class Squarecolor [0] xPos[61] yPos[61] dim[29] 
, Class Linecolor{3] xPos{68) yPos{6] dim[22] 
+ Class Circlecolor{1] xPos(7] yPos(88] dim(28] 
+ lass Squarecolor [0] xPos{S1] yPos[89] dim[9] 
+ Class Linecolor[3] xpos[78] ypos[98] dim[61] 
+ Class Circlecolor{1] xpPos[29] yPos[58] dim[16] 
, Class Squarecotor[6] xPos(40] yPos(11] dim(22] 
+ Class Linecolor [3] xpos[4] yPos(83] dim(6] 
, Class Circlecolor[1] xPos[75] yPos(10] dim(42} 


] 
AVI 


可 以 看 到 ，xPos、yPos 以 及 dim 的 值 都 被 成 功 地 保存 和 恢复 了 ， 但 是 对 static 信 息 的 读 取 却 
出 现 了 问题 。 所 有 读 回 的 颜色 应 该 都 是 “3”， 但 是 真实 情况 却 并 非 如 此 。Cirele 的 值 为 ! (定义 
为 RED)， 而 Square 的 值 为 0 ( 记 住 ， 它 们 是 在 构造 器 中 被 初始 化 的 )。 看 上 去 似乎 static 数 据 根 
本 没有 被 序列 化 ! 确实 如 此 一 一 尽管 Class 类 是 Serializable 的 ， 但 它 却 不 能 按 我 们 所 期 望 的 方式 
运行 。 所 以 假如 想 序列 化 static 值 ， 必 须 自己 动手 去 实现 。 

这 正 是 Line 中 的 serializeStaticState0 和 deserializeStaticState0 两 个 static 方 法 的 用 途 。 可 以 看 
到 ， 它 们 是 作为 存储 和 读 取 过 程 的 一 部 分 被 显 式 地 调用 的 。( 注 意 必 须 维护 写 人 序列 化 文件 和 从 
该 文件 中 读 回 的 顺序 。) 因此 ， 为 了 使 CADStatejava 正 确 运转 起 来 ， 我 们 必须 ; 

1) 为 几何 形状 添加 serializeStaticState0 和 deserializeStaticState0 。 

2) 移 除 ArrayList shapeTypes 以 及 与 之 有 关 的 所 有 代码 。 

3) 在 几何 形状 内 添加 对 新 的 序列 化 和 反 序列 化 还 原 静 态 方法 的 调用 。 

另 一 个 要 注意 的 问题 是 安全 ， 因 为 序列 化 也 会 将 private 数 据 保存 下 来 。 如 果 你 关心 安全 问 
题 ， 那 么 应 将 其 标记 成 transient。 但 是 这 之 后 ， 还 必须 设计 一 种 安全 的 保存 信息 的 方法 ， 以 便 
在 执行 恢复 时 可 以 复位 那些 private 变 量 。 

练习 30: (1) 按照 书 中 描述 ， 修 改 CADStatejava。 
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18.13 XML 


对 象 序列 化 的 一 个 重要 限制 是 它 只 是 Java 的 解决 方案 : 只 有 Java 程 序 才能 反 序 列 化 这 种 对 象 。 
一 种 更 具 互 操作 性 的 解决 方案 是 将 数据 转换 为 XML 格 式 ， 这 可 以 使 其 被 各 种 各 样 的 平台 和 语言 
使 用 。 

因为 XML 十 分 流行 ， 所 以 用 它 来 编程 时 的 各 种 选择 不 胜 枚 举 ， 包 括 随 JDK 发 布 的 
javax.xml.* 类 库 。 我 选择 使 用 Elliotte Rusty Harold 的 开源 XOM 类 库 (可 从 www.xom.nu 下 载 并 获 
得 文档 ) ， 因 为 它 看 起 来 最 简单 ， 同 时 也 是 最 直观 的 用 Java 产 生 和 修改 XML 的 方式 。 另 外 ， 
XOM 还 强调 了 XML 的 正确 性 。 

作为 一 个 示例 ， 假 设 有 一 个 Person 对 象 ， 它 包含 姓 和 名 ， 你 想 将 它们 序列 化 到 XML 中 。 下 
面 的 Person 类 有 一 个 getXML( 方 法 ， 它 使 用 XOM 来 产生 被 转换 为 XML 的 Element 对 象 的 Person 
数据 ， 还 有 一 个 构造 器 ， 接 受 Element 并 从 中 抽取 恰当 的 Person 数 据 (注意 ，XML 示 例 都 在 它们 
自己 的 子 且 录 中 ): 


/1: xml/Person.java 

// Use the XOM library to write and read XML 
// (Requires: nu.xom.Node; You must install 
// the XOM library from http://www.xom.nu } 
‘import nu. xom. 
import java.io.* 
import java.util.*; 





public class Person { 
private String first, last: 
public Person(String first, String last) { 
this. first = first; 
this.last = last; 


} 
// Produce an XML Element from this Person object: 


public Element getXML() { 
Element person = new Element("person") ; 
Element firstName = new Element("first"): 
firstName. appendChild(first) ; 
Element lastName = new Eleme 
LastName. appendChild(tast) ; 
person. appendChild(firstName) ; 
person. appendChild (lastName) ; 
return person; 
} 
// Constructor to restore a Person from an XML Element: 
public Person(Element person) { 
first= person.getFirstChildElement ("first") .getValue(); 
last = person. getFirstChildElement ("last") .getValue(); 





t("last"); 


} 
public String toString() { return first + " + last; } 
// Make it human-readable: a 
public static void 
format (OutputStream os, Document doc) throws Exception { 
Serializer serializer= new Serializer (os, "1S0-8859-1"); 
serializer.setIndent (4) ; 
serializer. setMaxLength(60) ; 
serializer.write(doc); 
serializer.flush(); 


public static void main(String[] args) throws Exception { 
List<Person> people = Arrays.asList( 
new Person("Dr. Bunsen”, "Honeydew"), 
new Person("Gonzo", “The Great” 
new Person("Phillip J.", "Fry") 
System.out.printin (people) ; 











Java VO 4% 587 





Element root = new Element (*péople"); 

for(Person p : people) 
root .appendChild(p.getXML()) ; 

Document doc = new Document (root); 

format (System.out, doc); 

format (new Buf feredOutputStream(new FileOutputStream( 
“People.xml")), doc): 


} 
} /* Output: 
[Dr. Bunsen Honeydew, Gonzo The Great, Phillip J. Fry] 
<?xml version="1.0" encoding="IS0-8859-1"?> 
<people> 
<person> 
<first>Dr. Bunsen</first> 
<last>Honeydew</Last> 
</person> 
<person> 
<first2Gonzo</first> 
<last>The Great</last> 
</person> 
<person> 
<firstoPhillip J.</first> 
<last>Fry</last> 
</person> 
</people> 
Wha 


XOM 的 方法 都 具有 相当 的 自 解释 性 ， 可 以 在 XOM 文 档 中 找到 它们 。XOM 还 包含 一 个 
Serializer 类 ， 你 可 以 在 format( 方 法 中 看 到 它 被 用 来 将 XML 转换 为 更 具 可 读 性 的 格式 。 如 果 只 
调用 toXMLO， 那 么 所 有 东西 都 会 混在 一 起 ， 因 此 Serializer 是 一 种 便利 工具 。 

从 XML 文 件 中 反 序列 化 Person 对 象 也 很 简单 : 


//: xml/People. java 

77 (Requires: nu.xom.Node; You must install 
// the XOM Library from http: //nww.xom.nu } 
// {RunFirst: Person} 

import nu.xom. 
import java.util.*; 





public class People extends ArrayList<Person> { 
public People(String fileName) throws Exception 《 
Document doc = new Builder() .build( fileName); 
Elements elements = 
doc. getRootELement() .getChildElements() ; 
for(int i = 0; i < elements.size(); i++) 
add(new Person(elements.get(i))); 


} 

public static void main(String{] args) throws Exception { 
People p = new People("People. xml"); 
System. out .printin(p); 


} h Output: 

[Dr. Bunsen Honeydew, Gonzo The Great, Phillip J. Fry) 

“H~ 

People 构 造 器 使 用 XOM 的 Builder.build0 方 法 打开 并 读 取 一 个 文件 ， 而 getChildElements0 方 
法 产生 了 一 个 Elements 列 表 (不 是 标准 的 Java List， 只 是 一 个 拥有 size0 和 get( 方 法 的 对 象 ， 因 
为 Harold 不 想 强制 人 们 使 用 Java SE5， 但 是 仍旧 希望 使 用 类 型 安全 的 容器 ) 。 在 这 个 列表 中 的 每 
个 Element 都 表示 一 个 Person 对 象 ， 因 此 它 可 以 传递 给 第 二 个 Person 构 造 器 。 注 意 ， 这 要 求 你 提 
前 知道 XML 文件 的 确切 结构 ， 但 是 这 经 常会 有 些 问题 。 如 果 文 件 结构 与 你 预期 的 结构 不 匹配 ， 
那么 XOM 将 抛 出 异常 。 对 你 来 说 ， 如 果 你 缺乏 有 关 将 来 的 XML 结构 的 信息 ， 那 么 就 有 可 能 会 编 
写 更 复杂 的 代码 去 探测 XML 文档 ， 而 不 是 只 对 其 做 出 假设 。 
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为 了 获取 这 些 示例 去 编译 它们 ， 你 必须 将 XOM 发 布 包 中 的 JAR 文 件 放置 到 你 的 类 路 径 中 。 

这 里 只 给 出 了 用 Java 和 XOM 类 库 进行 XML 编 程 的 简介 ， 更 详细 的 信息 可 以 浏览 www.xom.nu。 

练习 31: (2) 在 Person.java 和 People.java 中 添加 恰当 的 地 址 信息 。 

练习 32，(4) 使 用 Map<String,Integer> 和 net.mindview.util.TextFile 工 具 编 写 程序 ， 对 在 文 
件 中 出 现 的 单词 进行 计数 (使 用 NW+ 做 为 传递 给 TextFile 构 造 器 的 第 二 个 参数 ) 。 将 结果 存储 为 
XML 文件 。 


18.14 Preferences 


Preferences API 与 对 象 序列 化 相 比 ， 前 者 与 对 象 持久 性 更 密切 ， 因 为 它 可 以 自动 存储 和 读 取 
信息 。 不 过 ， 它 只 能 用 于 小 的 、 受 限 的 数据 集合 一 一 我 们 只 能 存储 基本 类 型 和 字符 串 ， 并 且 每 
个 字符 串 的 存储 长 度 不 能 超过 8K (不 是 很 小 ， 但 我 们 也 并 不 想 用 它 来 创建 任何 重要 的 东西 )。 顾 
名 思 义 ，Preferences API 用 于 存储 和 读 取 用 户 的 偏好 (preferences) 以 及 程序 配置 项 的 设置 。 

Preferences 是 一 个 键 -- 值 集合 (类似 映射 )， 存 储 在 一 个 节点 层次 结构 中 。 尽 管 节点 层次 结 
构 可 用 来 创建 更 为 复杂 的 结构 ， 但 通常 是 创建 以 你 的 类 名 命名 的 单一 节点 ， 然 后 将 信息 存储 于 


[1006] 共 中 。 下 面 是 一 个 简单 的 例子 : 


//: io/PreferencesDemo. java 
import java.util.prefs.*; 
import static net.mindview.util.Print.*; 


public class PreferencesDemo { 
public static void main(String[] args) throws Exception { 
Preferences prefs = Preferences 
-userNodeFor Package (PreferencesDemo.class) ; 
prefs.put("Location", "Oz"); 
prefs.put("Footwear", “Ruby Slippers"); 
prefs.putInt("Companions", 4): 
prefs.putBoolean("Are there witches?", true); 
int usageCount = prefs.getInt("UsageCount", 0); 
usageCount++; 
prefs.putInt("UsageCount", usageCount) ; 
for(String key : prefs.keys()) 
print(key + ": "+ prefs.get (key, null)): 
// You must always provide a default value: 
print("How many companions does Dorothy have? * + 
prefs.getInt("Companions”, 0)); 


} 
} /* Output: (Sample) 
Location: 0z 
Footwear: Ruby Slippers 
Companions: 4 
Are there witches?: true 
UsageCount: 53 
How many companions does Dorothy have? 4 
Whi 


这 里 用 的 是 userNodeForPackage0， 但 我 们 也 可 以 选择 用 systemNodeForPackage0， 虽 然 可 
以 任意 选择 ， 但 最 好 将 “user” 用 于 个 别 用 户 的 偏好 ， 将 “system” 用 于 通用 的 安装 配置 。 因 为 
main0) 是 静态 的 ， 因 此 PreferencesDemo.class 可 以 用 来 标识 节点 ， 但 是 在 非 静态 方法 内 部 ， 我 们 
通常 使 用 getClassO0。 尽 管 我 们 不 一 定 非 要 把 当前 的 类 作为 节点 标识 符 ， 但 这 仍 不 失 为 一 种 很 有 
用 的 方法 。 

一 县 我 们 创建 了 节点 ， 就 可 以 用 它 来 加 载 或 者 读 取 数 据 了 。 在 这 个 例子 中 ， 向 节点 载 信 了 
各 种 不 同类 型 的 数据 项 ， 然 后 获取 其 keysO0。 它 们 是 以 String[] 的 形式 返回 的 ， 如 果 你 习惯 于 
keys0 属 于 集合 类 库 ， 那 么 这 个 返回 结果 可 能 并 不 是 你 所 期 望 的 。 注 意 get0 的 第 二 个 参数 ， 如 果 
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某 个 关键 字 下 没有 任何 条 目 ， 那 么 这 个 参数 就 是 所 产生 的 默认 值 。 当 在 一 个 关键 字 集合 内 迭代 
时 ， 我 们 总 要 确信 条 目 是 存在 的 ， 因 此 用 muil 作 为 默认 值 是 安全 的 ， 但 是 通常 我 们 会 获得 一 个 具 
名 的 关键 字 ， 就 像 下 面 这 条 语句 : 

prefs .getInt("Companions"，6)): 

在 通常 情况 下 ， 我 们 希望 提供 一 个 合理 的 默认 值 。 实 际 上 ， 典 型 的 习惯 用 法 可 见 下 面 几 行 : 


int usageCount = prefs.getInt("UsageCount”, 0); 
usageCount++; 
prefs.putInt("UsageCount™, usageCount): 


这 样 ， 在 我 们 第 一 次 运行 程序 时 ，UsageCount 的 值 是 0， 但 在 随后 引用 中 ， 它 将 会 是 非 零 值 。 

在 我 们 运行 PreferencesDemo.javal 时 ， 会 发 现 每 次 运行 程序 时 ，UsageCount 的 值 都 会 增加 1， 
但 是 数据 存储 到 哪里 了 呢 ? 在 程序 第 一 次 运行 之 后 ， 并 没有 出 现任 何 本 地 文件 。Preferences API 
利用 合适 的 系统 资源 完成 了 这 个 任务 ， 并 且 这 些 资源 会 随 操作 系统 不 同 而 不 同 。 例 如 在 Windows 
里 ,就 使 用 注册 表 (因为 它 已 经 有 “ 键 值 对 ”这 样 的 节点 对 层次 结构 了 ) 。 但 是 最 重要 的 一 点 是 ， 
它 已 经 神奇 般 地 为 我 们 存储 了 信息 ， 所 以 我 们 不 必 担 心 不 同 的 操作 系统 是 怎么 运作 的 。 

还 有 更 多 的 Preferences API， 参 阅 JDK 文 档 可 很 容易 地 理解 更 深 的 细节 。 

练习 33，(2) 编写 一 个 程序 ， 显 示 被 称 为 “ 基 目 录 ” 的 目录 中 的 当前 值 ， 并 将 其 改编 为 你 的 
值 。 使 用 Preferences API 来 存储 这 个 值 。 


18.15 总 结 


Java IO 流 类 库 的 确 能 满足 我 们 的 基本 需求 : 我 们 可 以 通过 控制 台 、 文 件 、 内 存 块 ， 甚 至 因 
特 网 进行 读 写 。 通 过 继承 ， 我 们 可 以 创建 新 类 型 的 输入 和 输出 对 象 。 并 且 通 过 重新 定义 
toString() 方 法 ， 我 们 甚至 可 以 对 流 接受 的 对 象 类 型 进行 简单 扩充 。 当 我 们 向 一 个 期 望 收 到 字符 
串 的 方法 传送 一 个 对 象 时 ， 会 自动 调用 toString0 方 法 (这 是 Java 有 限 的 自动 类 型 转换 功能 ) 。 

在 LO 流 类 库 的 文档 和 设计 中 ， 仍 留 有 一 些 没有 解决 的 问题 。 例 如 ， 当 我 们 打开 一 个 文件 用 
于 输出 时 ， 我 们 可 以 指定 一 旦 试图 覆盖 该 文件 就 锰 出 一 个 异常 一 有 的 编程 系统 允许 我 们 自行 指 
定 想 要 打开 的 输出 文件 ， 只 要 它 尚 不 存在 。 在 Java 中 ， 我 们 似乎 应 该 使 用 一 个 File 对 象 来 判断 某 
个 文件 是 否 存在 ， 因 为 如 果 我 们 以 FileOutputStream 或 者 FileWriter 打 开 ， 那 么 它 肯定 会 被 覆盖 。 

VO 流 类 库 使 我 们 喜忧参半 。 它 确实 能 做 许多 事情 ， 而 且 具 有 可 移植 性 。 但 是 如 果 我 们 没有 
理解 “装饰 器 ”模式 ， 那 么 这 种 设计 就 不 是 很 直觉 因此， 在 学 习 和 传授 它 的 过 程 中 ， 需 要 额 
外 的 开销 。 而 且 它 并 不 完善 ， 例 如 ， 我 应 该 不 必 去 写 像 TextFile 这 样 的 应 用 (新 的 Java SE5 的 
PrintWriter 向 正确 的 方向 迈进 了 一 步 ， 但 是 它 只 是 一 个 部 分 的 解决 方案 )。 在 Java SE5 中 有 一 个 巨 
大 的 改进 : 他 们 最 终 添加 了 输出 格式 化 ， 而 事实 上 其 他 所 有 语言 的 /O 包 都 提供 这 种 支持 。 

一 旦 我 们 理解 了 装饰 器 模式 ， 并 开始 在 某 些 情况 下 使 用 该 类 库 以 利用 其 提供 的 灵活 性 ， 那 
么 你 就 开始 从 这 个 设计 中 受益 了 。 到 那个 时 候 ， 为 此 额外 多 写 几 行 代码 的 开销 应 该 不 至 于 使 人 
觉得 太 麻 烦 。 

所 选 习题 的 答案 都 可 以 在 名 为 The Thinking in Java Annotated Solution Guide 的 电子 文档 中 找 
到 ， 读 者 可 以 从 www.MindView.net 处 购买 此 文档 。 
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第 19 章 枚 举 类 型 


关键 字 enum 可 以 将 一 组 具名 的 值 的 有 限 集合 创建 为 一 种 新 的 类 型 ， 而 这 些 具名 的 值 可 以 作 
为 常规 的 程序 组 件 使 用 。 这 是 一 种 非常 有 用 的 功能 。 

在 第 5 章 结束 的 时 候 ， 我 们 已 经 简单 地 介绍 了 枚 举 的 概念 。 现 在 ， 你 对 Java 已 经 有 了 更 深刻 
的 理解 ， 因 此 可 以 更 深入 地 学 习 Java SE5 中 的 枚 举 了 。 你 将 在 本 章 中 看 到 ， 使 用 enum 可 以 做 很 
多 有 趣 的 事情 ， 同 时 ， 我 们 也 会 深入 其 他 的 Java 特 性 ， 例 如 泛 型 和 反射 。 在 这 个 过 程 中 ， 我 们 
还 将 学 习 一 些 设计 模式 。 


19.1 基本 enum 特 性 


我 们 已 经 在 第 5 章 看 到 ， 调 用 enum 的 values0 方 法 ， 可 以 遍历 enum 实 例 。values0 方 法 返回 
enum 实 例 的 数组 ， 而 且 该 数组 中 的 元 素 严格 保持 其 在 enum 中 声明 时 的 顺序 ， 因 此 你 可 以 在 循 
环 中 使 用 values0 返 回 的 数组 。 

创建 enum 时， 编译 器 会 为 你 生成 一 个 相关 的 类 ， 这 个 类 继承 自 java.lang.Enum。 下 面 的 例 
子 演示 了 Enum 提 供 的 一 些 功能 : 


//: enumerated/EnumClass.java 
// Capabilities of the Enum class 
‘import static net.mindview.util.Print.*; 


enum Shrubbery { GROUND, CRAWLING, HANGING } 


public class EnumClass { 
public static void main(String(] args) { 

for(Shrubbery s : Shrubbery.values()) { 
print(s + " ordinal: " + s.ordinal()); 
printnb(s.compareTo(Shrubbery.CRAWLING) + " "); 
printnb(s.equals(Shrubbery.CRAWLING) + " "); 
print(s == Shrubbery.CRAWLING) ; 
print(s.getDeclaringClass()); 
print(s.name()); 
Print ("-------2n eee ee ee nn "ys 

} 

7/ Produce an enum value from a string name: 

for (String $ : "HANGING CRAWLING GROUND".split(” ")) { 
Shrubbery shrub = Enum.valueOf(Shrubbery.class, s); 
print (shrub); 

} 


} 
} /* Output: 
GROUND ordinal: @ 
-1 false false 
class Shrubbery 
GROUND 





CRAWLING ordinal: 1 
6 true true 

class Shrubbery 
CRAWLING 


© Joshua Bloch 为 撰写 此 章 提供 了 很 大 帮助 。 
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HANGING ordinal: 2 
1 false false 
class Shrubbery 
HANGING 


HANGING 
CRAWLING 
GROUND 
Ii~ 


ordinal0 方 法 返回 一 个 int 值 ， 这 是 每 个 enum 实 例 在 声明 时 的 次 序 ， 从 0 开始 。 可 以 使 用 == 来 
比较 enum 实 例 ， 编 译 器 会 自动 为 你 提供 equals0 和 hashCode0 方 法 。Enum 类 实现 了 Comparable 
接口 ， 所 以 它 具 有 compareTo0 方 法 。 同 时 ， 它 还 实现 了 Serializable 接 口 。 

如 果 在 enum 实 例 上 调用 getDeclaringClass0 方 法 ， 我 们 就 能 知道 其 所 属 的 enum 类 。 

name0 方 法 返回 enum 实 例 声明 时 的 名 字 ， 这 与 使 用 toString0 方 法 效果 相同 。valueOf0 是 在 
Enum 中 定义 的 static 方 法 ， 它 根据 给 定 的 名 字 返 回 相应 的 enum 实 例 ， 如 果 不 存在 给 定名 字 的 实 
例 ， 将 会 抛 出 异常 。 


19.1.1 将 静态 导入 用 于 enum 
先 看 一 看 第 5 章 中 Burritojava 的 另 一 个 版 本 : 


//: enumerated/Spiciness. java 
package enumerated; 


public enum Spiciness { 
NOT, MILD, MEDIUM, HOT, FLAMING 

} Mi~ 

//: enumerated/Burrito. java 


package enumerated; 
import static enumerated. Spiciness.*; 


public class Burrito { 
Spiciness degree; 
public Burrito(Spiciness degree) { this.degree = degree; } 
public String toString() { return "Burrito is "+ degree:} 
public static void main(Stringl) args) { 
System. out.printin(new Burrito(NOT)) ; 
System. out.printin(new Burrito(MEDIUM)) ; 
System. out.printIn(new Burrito(HOT) ); 


} 
} /* Output: 
Burrito is NOT 
Burrito is MEDIUM 
Burrito is HOT 
IAA 


使 用 static import 能 够 将 enum 实 例 的 标识 符 带 入 当前 的 命名 空间 ， 所 以 无 需 再 用 enum 类 型 
来 修饰 enum 实 例 。 这 是 一 个 好 的 想法 吗 ? 或 者 还 是 显 式 地 修饰 enum 实 例 更 好 ? 这 要 看 代码 的 
复杂 程度 了 。 编 译 器 可 以 确保 你 使 用 的 是 正确 的 类 型 ， 所 以 唯一 需要 担心 的 是 ， 使 用 静态 导入 
会 不 会 导致 你 的 代码 令 人 难以 理解 。 多 数 情况 下 ， 使 用 static import 还 是 有 好 处 的 ， 不 过 ， 程序 
员 还 是 应 该 对 具体 情况 进行 具体 分 析 。 

注意 ， 在 定义 enum 的 同一 个 文件 中 ， 这 种 技巧 无 法 使 用 ， 如 果 是 在 默认 包 中 定义 enum， 
这 种 技巧 也 无 法 使 用 〈 在 Sun 内 部 对 这 一 点 显然 也 有 不 同意 见 )。 


19.2 向 enum 中 添加 新 方法 
除了 不 能 继承 自 一 个 enum 之 外 ， 我 们 基本 上 可 以 将 enum 看 作 一 个 常规 的 类 。 也 就 是 说 ， 
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我 们 可 以 向 enum 中 添加 方法 。enum 甚 至 可 以 有 main0 方 法 。 

一 般 来 说 ， 我 们 希望 每 个 枚 举 实例 能 够 返回 对 自身 的 描述 ， 而 不 仅仅 只 是 默认 的 toString0 
实现 ， 这 只 能 返回 枚 举 实例 的 名 字 。 为 此 ， 你 可 以 提供 一 个 构造 器 ， 专 门 负责 处 理 这 个 额外 的 
信息 ， 然 后 添加 一 个 方法 ， 返 回 这 个 描述 信息 。 看 一 看 下 面 的 示例 ; 


//: enumerated/OzWitch. java 
// The witches in the land of Oz. 
import static net.mindview.util.Print.*; 


public enum OzWitch { 
// Instances must be defined first, before methods: 
WEST("Miss Gulch, aka the Wicked Witch of the West"), 
NORTH("Glinda, the Good Witch of the North"), 
EAST("Wicked Witch of the East, wearer of the Ruby " + 
"Slippers, crushed by Dorothy's house"), 
SOUTH("Good by inference, but missing"); 
private String description: 
// Constructor must be package or private access: 
private OzWitch(String description) { 
this.description = description: 
} 
public String getDescription() { return description; } 
public static void main(String{] args) { 
for(OzWitch witch : OzWitch.values()) 
print(witch + ": * + witch.getDescription()); 
} 
} /* Output: 
WEST: Miss Gulch, aka the Wicked Witch of the West 
NORTH: Glinda, the Good Witch of the North 
EAST: Wicked Witch of the East, wearer of the Ruby 
Slippers, crushed by Dorothy's house 
SOUTH: Good by inference, but missing 
Wha 


注意 ， 如 果 你 打算 定义 自己 的 方法 ， 那 么 必须 在 enum 实 例 序列 的 最 后 添加 一 个 分 号 。 同 时 ， 
Java 要 求 你 必须 先 定义 enum 实 例 。 如 果 在 定义 enum 实 例 之 前 定义 了 任何 方法 或 属性 ， 那 么 在 纺 
译 时 就 会 得 到 错误 信息 。 

enum 中 的 构造 器 与 方法 和 普通 的 类 没有 区 别 ， 因 为 除了 有 少许 限制 之 外 ，enum 就 是 一 个 
普通 的 类 。 所 以 ， 我 们 可 以 使 用 enum 做 许多 事情 (虽然 ， 我 们 一 般 只 使 用 普通 的 枚 举 类 型 )。 

在 这 个 例子 中 ， 虽 然 我 们 有 意识 地 将 enum 的 构造 器 声明 为 private， 但 对 于 它 的 可 访问 性 而 
言 ， 其 实 并 没有 什么 变化 ， 因 为 即使 不 声明 为 private) 我 们 只 能 在 enum 定 义 的 内 部 使 用 其 构 
造 器 创建 enum 实 例 。 一 旦 enum 的 定义 结束 ， 编 译 器 就 不 允许 我 们 再 使 用 其 构造 器 来 创建 任何 
实例 了 。 
19.2.1 覆盖 enum 的 方法 

覆盖 toSring0 方 法 ， 给 我 们 提供 了 另 一 种 方式 来 为 枚 举 实例 生成 不 同 的 字符 串 描述 信息 。 
在 下 面 的 示例 中 ,我们 使 用 的 就 是 实例 的 名 字 ， 不 过 我 们 希望 改变 其 格式 。 覆盖 enum 的 
toSring0 方 法 与 覆盖 一 般 类 的 方法 没有 区 别 : 


//: enumerated/SpaceShip. java 
public enum SpaceShip { 
SCOUT, CARGO, TRANSPORT, CRUISER, BATTLESHIP, MOTHERSHIP; 
public String toString() { 
String id = name(): 
String lower = id.substring(1).toLowerCase(); 
return id.charAt(@) + lower; 
} 
public static void main(String(] args) { 
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for(SpaceShip s : values()) { i 
System.out.printin(s); 
} 


} 
~ ) /* Output: 
Scout 
Cargo 
Transport 
Cruiser 
Battleship 
Mothership 
711:~ 


toString( 方 法 通过 调用 name() 方 法 取得 SpaceShip 的 名 字 ， 然 后 将 其 修改 为 只 有 首 字母 大 写 的 
格式 。 


19.3 switch 语句 中 的 enum 


在 switch 中 使 用 enum， 是 enum 提 供 的 一 项 非常 便利 的 功能 。 一 般 来 说 ， 在 switch 中 只 能 使 
用 整数 值 ， 而 枚 举 实例 天 生 就 具备 整数 值 的 次 序 ， 并 且 可 以 通过 ordinal0 方 法 取得 其 次 序 ( 显 
然 编译 器 帮 我 们 做 了 类 似 的 工作 )， 因 此 我 们 可 以 在 switeh 语 句 中 使 用 enum。 

虽然 一 般 情况 下 我 们 必须 使 用 enum 类 型 来 修饰 一 个 enum 实 例 ， 但 是 在 case 语 句 中 却 不 必 如 
此 。 下 面 的 例子 使 用 enum 构 造 了 一 个 小 型 状态 机 : 


/1/: enumerated/TrafficLight.java 
// Enums in switch statements. 
import static net.mindview.util.Print.*; 


// Define an enum type: 
enum Signal { GREEN, YELLOW, RED. } 


public class Trafficlight ( 
Signal cotor = Signat.RED; 
public void change() { 
switch(color) { 
// Note that you don't have to say Signal.RED 
// in the case statement: 
case RED: color = Signal.GREEN; 


break; 
case GREEN: color = Signal.YELLOW; 
break; 
case YELLOW: color = Signal.RED; 
break; - 


} 


} 
public String toString() { 
return "The traffic light is ”+ color; 


} 
public static void main(String{] args) { 
TrafficLight t = new TrafficLight(); 
for(int 1 = @; 1 <7; i++) { 
print(t); 
t.change(); 
} 


} 
} 7/* Output: 
The traffic light 1s RED 
The traffic light is GREEN 
The traffic Light is YELLOW 
The traffic light is RED 
The traffic light is GREEN 
The traffic light is YELLOW 
The traffic light is RED 
Wh 
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编译 器 并 没有 抱怨 switch 中 没有 default 语 句 ， 但 这 并 不 是 因为 每 一 个 Signal 都 有 对 应 的 case 


语句 。 如 果 你 注释 掉 其 中 的 某 个 case 语 句 ， 编 译 器 同样 不 会 抱怨 什么 。 这 意味 着 ， 你 必须 确保 


自己 覆盖 了 所 有 的 分 支 。 但 是 ， 如 果 在 case 语 句 中 调用 return， 那 么 编译 器 就 会 抱怨 缺少 default 


语 名 了。 这 与 是 否 覆 盖 了 enum 的 所 有 实例 无 关 。 


练习 1，(2) 修改 TrafficLightjava， 使 用 static import， 使 之 无 需 用 enmm 类 型 修饰 其 实例 。 


19.4 values() 的 神秘 之 处 


前 面 已 经 提 和 到， 编译 器 为 你 创建 的 enum 类 都 继承 自 Enum 类 。 然 而 ， 如 果 你 研究 一 下 Enum 
类 就 会 发 现 ， 它 并 没有 values0 方 法 。 可 我 们 明明 已 经 用 过 该 方法 了 ， 难 道 存 在 某 种 “隐藏 的 ” 
方法 吗 ? 我 们 可 以 利用 反射 机 制 编写 一 个 简单 的 程序 ， 来 查看 其 中 的 究竟 : 


//: enumerated/Reflection. java 


// Analyzing enums using reflection. 
import java.lang.reflect.*; 

import java.util 
import net.mindview.util.*; 
import static net.mindview.util.Print,*; 








enum Explore { HERE, THERE } 


public class Reflection { 
public static Set<String> analyze(Class<?> srwaclass) { 
print("----- Analyzing ”+ enumClass + * ---- 
print("Interfaces:"); 
for (Type t : enumClass.getGenericInterfaces()) 
print(t) 
print("Base: ”+ enumClass.getSuperclass()); 
. print("Methods: "); 
Set<String> methods = new TreeSet<String>(); 
for (Method m : enumClass.getMethods()) 
methods .add(m.getName()); 
print (methods) ; 
return methods; 
} 
public static void main(Stringl) args) { 
Set<String> exploreMethods = analyze(Explore.class): 
Set<String> enumMethods = analyze(Enum.class); * 
print("Explore.containsAll (Enum)? ”+ 
exploreMethods .containsAll(enumMethods)) ; 
printnb("Explore. removeAll (Enum) = "); 
exploreMethods . removeAll (enumMethods) ; 
print (exploreMethods) ; 
// Decompile the code for the enum: 
OSExecute.command("javap Explore") ; 








} 
} /* Output: 
----- Analyzing class Explore ----- 
Interfaces: 
Base: class java. lang. Enum 
Methods: 
[compareTo, equals, getClass, getDeclaringClass, hashCode, 
name, notify, notifyAll, ordinal, toString, valueOf, 
values, wait] 
=--- Analyzing class java.lang.Enum ----- 
Interfaces: 
java. lang.Comparable<e> 
interface java.io.Serializable 
Base: class java.lang.Object 
Methods: 
[compareTo, equals, getClass, getDeclaringClass, hashCode, 
name, notify, notifyAll, ordinal, toString. valueOf, wait] 
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Explore.containsAll (Enum)? true 
Explore. removeAil(Enum): [values] 
Compiled from "Reflection. java" 
final class Explore extends java. lang.Enum{ 
public static final Explore HERE: 
public static final Explore THERE; 
public static final Explore[] values(); i 
public static Explore valueOf (java.lang.String); 
static (}: 


} 

hm 

答案 是 ，values0 是 由 编译 器 添加 的 static 方 法 。 可 以 看 出 ， 在 创建 Explore 的 过 程 中 ， 编 译 
器 还 为 其 添加 了 valueOf0 方 法 。 这 可 能 有 点 令 人 迷惑 ，Enum 类 不 是 已 经 有 valueOf0 方 法 了 吗 。 
不 过 Enum 中 的 valueOf0 方 法 需要 两 个 参数 ， 而 这 个 新 增 的 方法 只 需 一 个 参数 。 由 于 这 里 使 用 的 
Set 只 存储 方法 的 名 字 ， 而 不 考虑 方法 的 签名 ， 所 以 在 调用 Explore.removeAll(Enum) 之 后 ， 就 只 
剩 下 [values] 了 。 

从 最 后 的 输出 中 可 以 看 到 ， 编 译 器 将 Explore 标 记 为 final 类 ， 所 以 无 法 继承 自 enum。 其 中 还 
有 一 个 static 的 初始 化 子 句 ， 稍 后 我 们 将 学 习 如 何 重 定义 该 句 。 

由 于 擦 除 效应 (在 第 15 章 中 介绍 过 ) ， 反 编译 无 法 得 到 Enum 的 完整 信息 ， 所 以 它 展示 的 
Explore 的 父 类 只 是 一 个 原始 的 Enum， 而 非 事实 上 的 Enum<Explore>。 

由 于 values0 方 法 是 由 编译 器 插入 到 enum 定 义 中 的 static 方 法 ， 所 以 ， 如 果 你 将 enum 实 例 向 上 
转型 为 Enum， 那 么 values0 方 法 就 不 可 访问 了 。 不 过 ， 在 Class 中 有 一 个 getEnumConstants0 方 法 ， 
所 以 即便 Enum 接 口中 没有 values0 方 法 ， 我 们 仍然 可 以 通过 Class 对 象 取得 所 有 enum 实 例 : 


//: enumerated/UpcastEnum. java 
// No values() method if you upcast an enum 


enum Search { HITHER, YON } 


Public class UpcastEnum ( 
public static void main(String{] args) { 
Search[] vals = Search. values(); 
Enum e = Search.HITHER; // Upcast 
// e.values(); // No values() in Enum 
for(Enum en : e.getClass().getEnumConstants()) 
System. out.printin(en): 


} 
} /* Output: 
HITHER 
YON 
Whim 


因为 getEnumConstants0 是 Class 上 的 方法 ， 所 以 你 甚至 可 以 对 不 是 枚 举 的 类 调用 此 方法 ， 
/1/: enumerated/NonEnum.java 


public class NonEnum { 
public static void main(String{] args) { 
Class<Integer> intClass = Integer.class: 
try { 
for (Object en : intClass.getEnumConstants()) 
System.out.printin(en) ; 
} catch(Exception e) { 
System.out.printin(e); 
} 


} 
} /* Output: 
java. lang.Nul1PointerException 
“N~ 
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只 不 过 ， 此 时 该 方法 返回 null， 所 以 当 你 试图 使 用 其 返回 的 结果 时 会 发 生 异 常 。 
19.5 实现 ,而 非 继承 


我 们 已 经 知道 ， 所 有 的 enum 都 继承 自 java.lang.Enum 类 。 由 于 Java 不 支持 多 重 继承 ， 所 以 
你 的 enum 不 能 再 继承 其 他 类 : 


enum NotPossible extends Pet { ... // Won't work 
然而 ， 在 我 们 创建 一 个 新 的 enum 时 ， 可 以 同时 实现 一 个 或 多 个 接口 : ` 


//: enumerated/cartoons/EnumImplementation. java ‘ 
// An enum can implement an interface 
package enumerated. cartoons: 
import java.util.*; 
1020] import net.mindview.util.*; 
enum CartoonCharacter 
implements Generator<CartoonCharacter> { 
SLAPPY, SPANKY, PUNCHY, SILLY, BOUNCY, NUTTY, BOB; 
private Random rand = new Random(47) ; 
public CartoonCharacter next() 《 
return values() (rand.nextint (vatues().length)]; 





} 


public class EnumImplementation { 

public static <T> void printNext(Generator<T> rg) { 
System.out.print(rg.next() +", "); 

} 

public static void main(String[J args) { 
// Choose any instance: 
CartoonCharacter cc = CartoonCharacter .BOB; 
for(int i = 0; i < 10; i++) 

printNext (cc) ; 


} Output: 

BOB, PUNCHY, BOB, SPANKY, NUTTY, PUNCHY, SLAPPY, NUTTY, 

NUTTY, SLAPPY, 

i~ 
这 个 结果 有 点 奇 保 ， 不 过 你 必须 要 有 一 个 enum 实 例 才 能 调用 其 上 的 方法 。 现 在 ， 在 任何 接受 
Generator 参 数 的 方法 中 ， 例 如 printNext0， 都 可 以 使 用 CartoonCharacter。 

练习 2，(2) 修改 上 例 ， 编 写 一 个 static next0 方 法 取代 实现 Generater 接 口 。 对 比 这 两 种 方式 ， 
各 自 有 什么 优 缺 点 。 


19.6 随机 选取 


就 像 你 在 CartoonCharacternext(0 中 看 到 的 那样 ， 本 章 中 的 很 多 示例 都 需要 从 enum 实 例 
中 进行 随机 选择 。 我 们 可 以 利用 泛 型 ， 从 而 使 得 这 个 工作 更 一 般 化 ， 并 将 其 加 入 到 我 们 的 工 
具 库 中 。 


//: net/mindview/utit/Enums.java 
package net.mindview.util; 
import java.util.*; 


public class Enums { 
private static Random rand = new Random(47) ; 











1021 public static <T extends Enum<T>> T random(Class<T> ec) { 





return random(ec.getEnumConstants()); 


public static <T> T random(T{} values) { 
return values [rand.nextInt (values. length) ] : 
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} 
} Mi~ 


古怪 的 语法 <T extends Enum<T>> 表 示 T 是 一 个 enum 实 例 。 而 将 Class<T> 作 为 参数 的 话 ， 
我 们 就 可 以 利用 Class 对 象 得 到 enum 实 例 的 数组 了 。 重 载 后 的 random() 方 法 只 需 使 用 TD 作为 参 
数 ， 因 为 它 并 不 会 调用 Enum 上 的 任何 操作 ， 它 只 需 从 数组 中 随机 选择 一 个 元 素 即 可 。 这 样 ， 最 
终 的 返回 类 型 正 是 enum 的 类 型 。 

下 面 是 random0 方 法 的 一 个 简单 示例 : - 


//: enumerated/RandomTest .java 
import net,mindview.utit.*: 


enum Activity { SITTING, LYING, STANDING, HOPPING, 
RUNNING, DODGING, JUMPING, FALLING, FLYING } 
public class RandomTest { 
public static void main(String] args) { 
for(int 1 = @; 1 < 20; i++) 5 
System.out.print (Enums. random(Activity.class) +" "); 


} 
} /* Output: 
STANDING FLYING RUNNING STANDING RUNNING STANDING LYING . 
DODGING SITTING RUNNING HOPPING HOPPING HOPPING RUNNING 7 
STANDING LYING FALLING RUNNING FLYING LYING i a 
“Wham 


虽然 Enum 只 是 一 个 相当 短小 的 类 ， 但 是 在 本 章 中 你 会 发 现 ， 它 能 消除 很 多 重复 的 代码 。 重复 总 
会 制造 麻烦 ， 因 此 消除 重复 总 是 有 益处 的 。 ， 


19.7 使 用 接口 组 织 枚 举 ， 


无 法 从 enum 继 承 子 类 有 时 很 令 人 诅 形 。 这 种 需求 有 时 源 自我 们 希望 扩展 原 enum 中 的 元 素 ， 
有 时 是 因为 我 们 希望 使 用 子 类 将 一 个 enum 中 的 元 素 进行 分 组 。 

在 一 个 接口 的 内 部 ， 创 建 实现 该 接口 的 枚 举 ， 以 此 将 元 素 进行 分 组 ， 可 以 达到 将 枚 举 元 素 
分 类 组 织 的 目的 。 举 例 来 说 ， 假 设 你 想 用 enum 来 表示 不 同类 别 的 食物 ， 同 时 还 希望 每 个 enum 
元 素 仍然 保持 Food 类 型 。 那 可 以 这 样 实现 : 


//: enumerated/menu/Food. java 
// Subcategorization of enums within interfaces. 
package enumerated.menu; 


`a 


public interface Food { 
enum Appetizer implements Food { ier 
SALAD, SOUP, SPRING_ROLLS; 3 ; ， 


} 

enum MainCourse implements Food { 
LASAGNE, BURRITO, PAD_THAT, 
LENTILS, HUMMOUS, VINDALOO; 

} 

enum Dessert implements Food { 
TIRAMISU, GELATO, BLACK_FOREST_CAKE, 
FRUIT, CREME_CARAMEL; 

} 

enum Coffee implements Food { 
BLACK COFFEE, DECAF_COFFEE, ESPRESSO, 
LATTE, CAPPUCCINO, TEA, HERB_TEA; 


} 
} Mi~ 
对 于 enum 而 言 ， 实 现 接 口 是 使 其 子 类 化 的 唯一 办 法 ， 所 以 嵌入 在 Food 中 的 每 个 enum 都 实 
现 了 Food 接 口 。 现 在 ， 在 下 面 的 程序 中 ， 我 们 可 以 说 “所 有 东西 都 是 某 种 类 型 的 Food : 


1022| 
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//: enumerated/menu/Type0fFood. java 
package enumerated. menu; 
import static enumerated.menu. Food. *; 


public class TypeOfFood { 
public static void main(String{] args) { 
Food food = Appetizer. SALAD; 
food = MainCourse.LASAGNE: 
food = Dessert. GELATO; 
food = Coffee. CAPPUCCINO: 





} 
yh 


如 果 enum 类 型 实现 了 Food 接 口 ， 那 么 我 们 就 可 以 将 其 实例 向 上 转型 为 Food， 所 以 上 例 中 的 所 有 
东西 都 是 Food 。 


然而 ， 当 你 需要 与 一 大 堆 类 型 打交道 时 ， 接 口 就 不 如 enum 好 用 了 。 例 如 ， 如 果 你 想 创建 一 


个 “ 枚 举 的 枚 举 "， 那 么 可 以 创建 一 个 新 的 enum， 然 后 用 其 实例 包装 Food 中 的 每 一 个 enum 类 ; 


//: enumerated/menu/Course.java 
Package enumerated.menu; 
‘import net.mindview.util.*; 


public enum Course { 

APPETIZER(Food. Appetizer.class), 

MAINCOURSE (Food. MainCourse. class), 

DESSERT(Food.Dessert.class), 

COFFEE (Food. Coffee.class) ; 

private Food{] values; 

private Course(Class<? extends Food> kind) { 
values = kind. getEnumConstants(); 


} 
public Food randomSelection() { 
return Enums.random(values) ; 


} 
} Mi~ 


在 上 面 的 程序 中 ， 每 一 个 Course 的 实例 都 将 其 对 应 的 Class 对 象 作为 构造 器 的 参数 。 通 过 


getEnumConstants0 方 法 ， 可 以 从 该 Class 对 象 中 取得 某 个 Food 子 类 的 所 有 enum 实 例 。 这 些 实例 
在 randomSelection0 中 被 用 到 。 因 此 ， 通 过 从 每 一 个 Course 实 例 中 随机 地 选择 一 个 Food， 我 们 
便 能 够 生成 一 份 菜单 : 


//: enumerated/menu/Meal.java 
package enumerated.menu; 


public class Meal { 
public static void main(Stringl] args) { 
for(int 1 = @; i < 5; i++) { 
for (Course course : Course.values()) { 
Food food = course. randomSelection(); 
System.out.printin(food) ; 


System. out. printin("--- 





} 
} /* Output: 
SPRING_ROLLS 
VINDALOO 
FRUIT 
DECAF_COFFEE 
SOUP 
VINDALOO ` 
FRUIT FS 
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TEA 

SALAD 

BURRITO 

FRUIT 

TEA 

SALAD 

BURRITO 
CREME_CARAMEL 
LATTE 


SOUP 

BURRITO 

TIRAMISU 

ESPRESSO $ 





VendingMachinejava 中 ， 我 们 会 看 到 另 一 种 组 织 枚 举 实例 的 方式 ， 但 其 也 有 一 些 其 他 的 限制 。 
此 外 ， 还 有 一 种 更 简洁 的 管理 枚 举 的 办 法 ， 就 是 将 一 个 enum 幅 套 在 另 一 个 enum 内 。 就 像 
这 样 : 


//: enumerated/SecurityCategory.java 
// More succinct subcategorization of enums. 
import net.mindview.util.*; 


enum SecurityCategory { 

STOCK(Security.Stock.class), BOND(Security.Bond.class); 

Security[] values: 

SecurityCategory (Class<? extends Security> kind) { 
values = kind.getEnumConstants(); 

} 

interface Security { 
enum Stock implements Security { SHORT, LONG, MARGIN } 
enum Bond implements Security { MUNICIPAL, JUNK } 


} 
public Security randomSelection() { 
return Enums.random(values) ; 


$ 
public static void main(String[] args) { 
for(int 1 =@; 1 < 10; i++) { 
SecurityCategory category = 
Enums. random(SecurityCategory.class); 
System.out.printin(category + ": "+ 
category.randomSelection()); 
} 


条 
} /* Output: 
BOND: MUNICIPAL 
BOND: MUNICIPAL 
STOCK: MARGIN 
STOCK: MARGIN 


BOND: JUNK 
STOCK: SHORT 
STOCK: LONG 
STOCK: LONG 
BOND: MUNICIPAL 
BOND: JUNK 
Whim 


Security 接 口 的 作用 是 将 其 所 包含 的 enum 组 合成 一 个 公共 类 型 ， 这 一 点 是 有 必要 的 。 然 后 ， 
SecurityCategory 才 能 将 Security 中 的 enum 作 为 其 构造 器 的 参数 使 用 ， 以 起 到 组 织 的 效果 。 








[zg 
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如 果 我 们 将 这 种 方式 应 用 于 Food 的 例子 ， 结 果 应 该 这 样 : 


//: enumerated/menu/Meal2. java 
package enumerated.menu; 
import net.mindview.util.*; 


public enum Meal2 { 
APPETIZER(Food .Appetizer.class), 
MAINCOURSE (Food .MainCourse.class), 
DESSERT (Food. Dessert.class), 
COFFEE (Food. Coffee.class) ; 
private Food[] values; 
private Meal2(Class<? extends Food> kind) { 
values = kind. getEnumConstants(): 
} 
public interface Food { 
enum Appetizer implements Food { 
SALAD, SOUP, SPRING_ROLLS; 
} 
enum MainCourse implements Food { . 
LASAGNE, BURRITO, PAD_THAT, 
LENTILS, HUMMOUS, VINDALOO; 
Ey 
enum Dessert implements Food { 
TIRAMISU, GELATO, BLACK_FOREST_CAKE, 
FRUIT, CREME_CARAMEL; 
} 
enum Coffee implements Food { 
BLACK_COFFEE, DECAF_COFFEE, ESPRESSO, 
LATTE, CAPPUCCINO, TEA, HERB_TEA; 
} 


public Food randomSelection() {, 
return Enums.random(values) ; 


} 
public static void main(String[] args) { 
for(int 1 = 0; i <5; i++) { 
for(Meal2 meal : Meal2.values()) { 
Food food = meal.randomSelection(); 
System, out.printin(food) ; 


System.out.printin("---"); 


} I+ sane output as Meal. java *///:~ 
其 实 ， 这 仅仅 是 重新 组 织 了 一 下 代码 ， 不 过 多 数 情况 下 ， 这 种 方式 使 你 的 代码 具有 更 清晰 的 结构 。 

练习 3: (1) 向 Course.java 中 添加 一 个 新 的 Course， 证 明 它 在 Mealjava 中 能 正确 工作 。 

练习 4: (1) 针对 Meal2.java， 重 复 前 一 个 练习 。 

练习 5， (4) 修改 control/VowelsAndConsonants.java， 使 用 3 个 enum 类 型 : VOWEL, 
SOMETIMES_A_VOWEL， 以 及 CONSONANT。 其 中 的 enum 构 造 器 应 该 可 以 接受 属于 不 同类 
别 的 各 种 字母 。 提 示 : 使 用 可 变 参 数 。 要 记 住 ， 可 变 参 数 会 自动 为 你 创建 一 个 数组 。 

练习 6: (3) 试 比 较 以 下 两 种 方式 的 优 缺 点 : 第 一 : 将 Appetizer、MainCourse、Desert 和 
Coffee 嵌 人 在 Food 内 部 ， 第 二 : 将 它们 实现 为 单独 的 enum， 并 各 自 实 现 Food 接 口 。 


19.8 使 用 EnumSet 蔡 代 标志 


Set 是 一 种 集合 ， 只 能 向 其 中 添加 不 重复 的 对 象 。 当 然 ，enum 也 要 求 其 成 员 都 是 唯一 的 ， 所 
以 enumi 看 起 来 也 具有 集合 的 行为 。 不 过 ， 由 于 不 能 从 enum 中 删除 或 添加 元 素 ， 所 以 它 只 能 算 
是 不 太 有 用 的 集合 。Java SE5 引 入 EnumSet， 是 为 了 通过 enum 创 建 一 种 替代 品 ， 以 替代 传统 的 
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基于 int 的 “位 标志 "。 这 种 标志 可 以 用 来 表示 某 种 “ 开 / 关 ”信息 ， 不 过 ， 使 用 这 种 标志 ， 我 们 
最 终 操作 的 只 是 一 些 bit， 而 不 是 这 些 bit 想 要 表达 的 概念 很 容易 写 出 令 人 难以 理解 的 代码 。 

EnumSet 的 设计 充分 考虑 到 了 速度 因素 ， 因 为 它 必须 与 非常 高 效 的 bit 标 志 相 竞争 (其 操作 
与 HashSet 相 比 ， 非 常 地 快 )。 就 其 内 部 而 言 ， 它 (可 能 ) 就 是 将 一 个 long 值 作为 比特 向 量 ， 所 
以 EnumSet 非 常 快速 高 效 。 使 用 EnumSet 的 优点 是 ， 它 在 说 明 一 个 二 进 制 位 是 否 存在 时 ， 具有 
更 好 的 表达 能 力 ， 并 且 无 需 担 心性 能 。 

EnumSet 中 的 元 素 必须 来 自 一 个 enum。 下 面 的 enum 表 示 在 一 座 大 楼 中 ， 警 报 传感器 的 安放 
位 置 ; 

/1/: enumerated/AlarmPoints. java 

package enumerated; 

public enum AlarmPoints { 

STAIR1, STAIR2, LOBBY, OFFICE1, OFFICE2, OFFICE3, 


OFFICE4, BATHROOM, UTILITY, KITCHEN 
} Mi~ 


然后 ， 我 们 用 EnumSet 来 跟踪 报警 器 的 状态 : 





//: enumerated/EnumSets.java 
// Operations on EnumSets 
Package enumerated; 
import java.util.*; 
import static enumerated.AlarmPoints 
import static net.mindview.util.Prin 
public class EnumSets { 
public static void main(Stringl] args) { 
EnumSet<AlarmPoints> points ‘= 
EnumSet .noneOf (AlarmPoints.class); // Empty set 
points. add (BATHROOM) ; 
print (points) ; 
points. addAll(EnumSet .of (STAIR1, STAIR2, KITCHEN)); 
print (points) ; 
points = EnumSet.allOf(AlarmPoints.class); 
points. removeAll (EnumSet .of (STAIR1, STAIR2, KITCHEN); 
print(points); 
points. removeAll (EnumSet. range (OFFICE1, OFFICE4)) ; 
print(points) ; 
points = EnumSet .complementOf (points) ; 
print (points) ; 
} 
} /* Output: 
{BATHROOM} 
[STAIR1, STAIR2, BATHROOM, KITCHEN] 
[LOBBY, OFFICE1, OFFICE2, OFFICE3, OFFICE4, BATHROOM, 
UTILITY] 
(LOBBY, BATHROOM, UTILITY] 
[STAIR1, STAIR2, OFFICE1, OFFICE2, OFFICE3, OFFICE4, 
KITCHEN] 
Whim 


使 用 static import 可 以 简化 enum 常 量 的 使 用 。EnumSet 的 方法 的 名 字 都 相当 直观 ， 你 可 以 
查阅 ]DK 文 档 找到 其 完整 详细 的 措 述 。 如 果 仔细 研究 了 EnumSet 的 文档 ， 你 还 会 发 现 一 个 有 趣 
的 地 方 ，of0 方 法 被 重 载 了 很 多 次 ， 不 但 为 可 变数 量 参数 进行 了 重 载 ， 而 且 为 接收 2 至 5 个 显 式 的 
参数 的 情况 都 进行 了 重 载 。 这 也 从 侧面 表现 了 EnumSet 对 性 能 的 关注 。 因 为 ， 其 实 只 使 用 可 变 
参数 已 经 可 以 解决 整个 问题 了 ， 但 是 对 比 显 式 的 参数 ， 会 有 一 点 性 能 损失 。 采 用 现在 这 种 设计 ， 
当 你 只 使 用 2 到 5 个 参数 调用 of0 方 法 时 ， 你 可 以 调用 对 应 的 重 载 过 的 方法 (速度 稍 快 一 点 ) ， 而 
当 你 使 用 一 个 参数 或 多 过 5 个 参数 时 ， 你 调用 的 将 是 使 用 可 变 参 数 的 of0 方 法 。 注 意 ， 如 果 你 只 
使 用 一 个 参数 ， 编 译 器 并 不 会 构造 可 变 参 数 的 数组 ， 所 以 与 调用 只 有 一 个 参数 的 方法 相 比 ， 也 
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就 不 会 有 额外 的 性 能 损耗 。 

EnumSet 的 基础 是 long， 一 个 long 值 有 64 位 ， 而 一 个 enum 实 例 只 需 一 位 bit 表 示 其 是 否 存在 。 
也 就 是 说 ， 在 不 超过 一 个 long 的 表达 能 力 的 情况 下 ， 你 的 EnumSet 可 以 应 用 于 最 多 不 超过 64 个 
元 素 的 enum。 如 果 enum 超 过 了 64 个 元 素 会 发 生 什么 呢 ? 


//: enumerated/BigEnumSet. java 
import java.util.*; 


public class BigEnumSet { 

enum Big { AQ, Al, A2, A3, A4, AS, AG, A7, AB, A9, A19, 
A11, A12, A13, A14, A15, A16, A17, A18, A19, A20, A21, 
A22, A23, A24, A25, A26, A27, A28, A29, AZO, A31, A32, 
A33, A34, A35, A36, A37, A38, A39, A40, A41, A42, A43, 
A44, A45, A46, A47, A48, A49, AS, AS1, A52, A53, AS4, 
ASS, A56, A57, A58, A59, AGO, AGL, A62, A63, A64, AGS, 
A66, A67, A68, AGS, A70, A71, A72, A73, A74, A75 } 

public static void main(String[] args) { 
EnumSet<Big> bigEnumSet = EnumSet.allOf(Big.class); 
System. out. print 1n(bigEnumSet) ; 


} 

} /* Output: 
(A®, Al, A2, A3, A4, AS, AG, A7, A8, A9, A10, All, A12, 
A13, A14, , A16, A17, A18, A19, A20, A21, A22, A23, A24, 
A25, A26, » A28, A29, A30, A31, A32, A33, A34, A35, A36, 

+ A38. . A40, A41, A42, A43, A44, ASS, A46, A47, A48, 
. A52, A53, A54, ASS, A56, A57, A58, A59, A60, 
. A64, A65, A66, A67, A68, A69, A70, A71, A72, 






显然 ，EnumSet 可 以 应 用 于 多 过 64 个 元 素 的 enum， 所 以 我 猜测 ，Enum 会 在 必要 的 时 候 增 
加 一 个 long。 
练习 7，(3) 找到 EnumSet 的 源 代码 ， 解 释 其 工作 原理 。 


19.9 使 用 EnumMap 


EnumMap 是 一 种 特殊 的 Map ， 它 要 求 其 中 的 键 (key) 必须 来 自 一 个 enum。 由 于 enum 本 
身 的 限制 ， 所 以 EnumMap 在 内 部 可 由 数组 实现 。 因 此 EnumMap 的 速度 很 快 ， 我 们 可 以 放心 地 
使 用 enum 实 例 在 EnumMap 中 进行 查找 操作 。 不 过 ， 我 们 只 能 将 enum 的 实例 作为 键 来 调用 putO 
方法 ， 其 他 操作 与 使 用 一 般 的 Map 差 不 多 。 

下 面 的 例子 演示 了 命令 设计 模式 的 用 法 。 一 般 来 说 ， 命 令 模式 首先 需要 一 个 只 有 单一 方法 
的 接口 ， 然 后 从 该 接口 实现 具有 各 自 不 同 的 行为 的 多 个 子 类 。 接 下 来 ， 程 序 员 就 可 以 构造 命令 
对 象 ， 并 在 需要 的 时 候 使 用 它们 了 : 

/1/: enumerated/EnumMaps. java 

// Basics of EnumMaps. 

package enumerated; 

import java.util.*; 


import static enumerated.AlarmPoints.*; 
import static net.mindview.util.Print.*; 


interface Command { void action(); } 


public class EnumMaps { 
public static void main(String[] args) { 
EnumMap<AlarmPoints,Command> em = 
new EnumMap<AlarmPoints ,Command>(AlarmPoints.class) ; 
em.put (KITCHEN, new Command() { 
public void action() { print("Kitchen fire!"); } 
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D: 
em.put (BATHROOM, new Command() { 
public void action() { print("Bathroom alert!"); } 
D: 
for (Map. Entry<AlarmPoints,Command> e : em.entrySet()) { 
printnb(e.getKey() + ": "); 
e.getValue() .action(); 


} 

try { // If there's no value for a particular key: 
em. get (UTILITY) .action(); 

} catch(Exception e) { 
print(e); 

} 


} 
} /* Output: 
BATHROOM: Bathroom alert! 
KITCHEN: Kitchen fire! 
java. lang.NullPointerException 
Whim 


与 EnumSet 一 样 ，enum 实 例 定义 时 的 次 序 决定 了 其 在 EnumMap 中 的 顺序 。 

main0 方 法 的 最 后 部 分 说 明 ，enum 的 每 个 实例 作为 一 个 键 ， 总 是 存在 的 。 但 是 ， 如 果 你 没 
有 为 这 个 键 调 用 put0 方 法 来 存 人 相应 的 值 的 话 ， 其 对 应 的 值 就 是 null。 

与 常量 相关 的 方法 constant-specific methods 将 在 下 一 节 中 介绍 ) 相 比 ，EnumMap 有 一 个 
优点 ， 那 EnumMap 允 许 程序 员 改 变 值 对 象 ， 而 常量 相关 的 方法 在 编译 期 就 被 固定 了 。 

稍 后 你 会 看 到 ， 在 你 有 多 种 类 型 的 enum， 而 且 它们 之 间 存在 互 操作 的 情况 下 ,我们 可 以 用 
EnumMap 实 现 多 路 分 发 (multiple dispatching). 


19.10 常量 相关 的 方法 


Java 的 enum 有 一 个 非常 有 趣 的 特性 ， 即 它 允 许 程序 员 为 enum 实 例 编写 方法 ， 从 而 为 每 个 
enum 实 例 赋予 各 自 不 同 的 行为 。 要 实现 常量 相关 的 方法 ， 你 需要 为 enum 定 义 一 个 或 多 个 abstract 
方法 ， 然 后 为 每 个 enum 实 例 实现 该 抽象 方法 。 参 考 下 面 的 例子 : 


//: enumerated/ConstantSpecificMethod. java 
import java.util. *; 
import java.text.*; 


public enum ConstantSpecificMethod { 
DATE_TIME { 
String getInfo() { 
return 
DáteFormat.getDateInstance().format(new Date()): 
) 


}, 
CLASSPATH { 
String getInfo() { 
return System. getenv ("CLASSPATH"): 
} 
}, 
VERSION { 
String getInfo() { ; 
return System.getProperty("java.version"); 
} 
j: 
abstract String getInfo(); 
public static void main(String[] args) { 
for (ConstantSpecificMethod csm : values()) 
System.out.println(csm.getInfo()) : 
} 
} /* (Execute to see output) *///:~ 





[oa 
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通过 相应 的 enum 实 例 ， 我 们 可 以 调用 其 上 的 方法 。 这 通常 也 称 为 表 驱 动 的 代码 (table- 
driven code， 请 注意 它 与 前 面 提 到 的 命令 模式 的 相似 之 处 ) 。 

在 面向 对 象 的 程序 设计 中 ， 不 同 的 行为 与 不 同 的 类 关联 。 而 通过 常量 相关 的 方法 ， 每 个 
enum 实 例 可 以 具备 自己 独特 的 行为 ， 这 似乎 说 明 每 个 enum 实 例 就 像 一 个 独特 的 类 。 在 上 面 的 
例子 中 ，enum 实 例 似乎 被 当 作 其 “ 超 类 ”ConstantSpecificMethod 来 使 用 ， 在 调用 getInfo0 方 法 
时 ， 体现 出 多 态 的 行为 。 

然而 ，enum 实 例 与 类 的 相似 之 处 也 仅 限于 此 了 。 我 们 并 不 能 真 的 将 enum 实 例 作 为 一 个 类 
型 来 使 用 : 


//: enumerated/NotClasses. java 
71 (Exec: javap -c LikeClasses} 
import static net.mindview.util.Print.*; 


enum LikeClasses { 
WINKEN { void behavior() { print("Behaviorl"); } }, 
BLINKEN { void behavior() { print("Behavior2"); } }, 
NOD { void behavior() { print("Behavior3"); } }; 
abstract void behavior (); 

} 


public class. NotClasses { 

// void fi(LikeClasses.WINKEN instance) {} // Nope 
} /* Output: ~ 
Compiled from "NotClasses.java" 
abstract class LikeClasses extends java. lang. Enum{ 
public static final LikeClasses WINKEN; 


public static final LikeClasses BLINKEN; 

public static final LikeClasses NOD; 

Wh 

在 方法 f10 中 ， 编 译 器 不 允许 我 们 将 一 个 enum 实 例 当 作 class 类 型 。 如 果 我 们 分 析 一 下 编译 
器 生成 的 代码 ， 就 知道 这 种 行为 也 是 很 正常 的 。 因 为 每 个 enum 元 素 都 是 一 个 LikeClasses 类 型 的 
static final 实 例 。 

同时 ， 由 于 它们 是 static 实 例 ， 无 法 访问 外 部 类 的 非 static 元 素 或 方法 ， 所 以 对 于 内 部 的 
enum 的 实例 而 言 ， 其 行为 与 一 般 的 内 部 类 并 不 相同 。 

再 看 一 个 更 有 趣 的 关于 洗车 的 例子 。 每 个 顾客 在 洗车 时 ， 都 有 一 个 选择 菜单 ， 每 个 选择 对 
应 一 个 不 同 的 动作 。 可 以 将 一 个 常量 相关 的 方法 关联 到 一 个 选择 上 ， 再 使 用 一 个 EnumSet 来 保 
存 客户 的 选择 : 


//: enumerated/CarWash. java 
import java.util.*; 
import static net.mindview.util.Print.*; 


public class CarWash { 
public enum Cycle { 
UNDERBODY { 
void action() { print("Spraying the underbody"): } 


}. 
WHEELWASH { 

void action() { print("Washing the wheels"); } 
}, 
PREWASH { 

void action() { print("Loosening the dirt"); } 
}, 
BASIC { 

void action() { print("The basic wash"); } 
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}, 
HOTWAX { 
void action() { print("Applying hot wax 
}, 
RINSE { 
void action() { print("Rinsing"); } 





h 

BLOWDRY { 
void action() { print("Blowing dry"): > 
i 
abstract void action(); ' 

) 


EnumSet<Cycle> cycles = 
EnumSet .of (Cycle. BASIC, Cycle.RINSE) ; 
public void add(Cycle cycle) { cycles.add(cycle); } 
public void washCar() { 
for (Cycle c : cycles) 
c.action(); 


4 
public String toString() { return cycles.toString(); } 
public static void main(String{] args) { 
CarWash wash = new CarWash(); 
print(wash) ; 
wash .washCar () ; 
// Order of addition is unimportant: 
wash. add(Cycle.BLOWDRY) ; 
wash.add(Cycle.BLOWORY); // Duplicates ignored 
wash. add (Cycle. RINSE) ; 
wash. add (Cycle HOTWAX) ; 
print (wash); 
wash. washCar(); 


} 
} /* Output: 5 
(BASIC, RINSE} 
The basic wash 
Rinsing 
[BASIC, HOTWAX, RINSE, BLOWDRY] 
The basic wash 
Applying hot wax 
Rinsing 
Blowing dry 
Whim 


与 使 用 匿名 内 部 类 相 比较 ， 定 义 常 量 相关 方法 的 语法 更 高 效 、 简 洁 。 

这 个 例子 也 展示 了 EnumSet 了 一 些 特性 。 因 为 它 是 一 个 集合 ， 所 以 对 于 同一 个 元 素 而 言 ， 
只 能 出 现 一 次 ， 因 此 对 同一 个 参数 重复 地 调用 add0 方 法 会 被 忽略 掉 (这 是 正确 的 行为 ， 因 为 一 
个 bit 位 开关 只 能 “打开 ”一 次 )。 同 样 地 ， 向 EnumSet 添 加 enum 实 例 的 顺序 并 不 重要 ， 因 为 其 
输出 的 次 序 决定 于 enum 实 例 定义 时 的 次 序 。 

除了 实现 abstract 方 法 以 外 ， 程 序 员 是 否 可 以 覆盖 常量 相关 的 方法 呢 ? 答案 是 肯定 的 ， 参 考 
下 面 的 程序 ， 


//: enumerated/OverrideConstantSpecific.java 
import static net.mindview.util.Print.*; 








public enum OverrideConstantSpecific { 
NUT, BOLT, 
WASHER { 
void f() { print("Overridden method"); } 


}; 
void f() { print("default behavior"): } 
public static void main(String[] args) { 
for(OverrideConstantSpecific ocs : values()) { 
printnb(ocs + ": "); 
ocs.f(); 
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} 


} 
} /* Output: 
NUT: default behavior 
BOLT: default behavior 
WASHER: Overridden method 
Wha 


虽然 enum 有 某 些 限制 ， 但 是 一 般 而 言 ， 我 们 还 是 可 以 将 其 看 作 是 类 。 


19.10.1 使 用 enum 的 职责 链 

在 职责 链 (Chain of Responsibility) 设计 模式 中 , 程序 员 以 多 种 不 同 的 方式 来 解决 一 个 问题 ， 
然后 将 它们 链接 在 一 起 。 当 一 个 请 求 到 来 时 ， 它 遍历 这 个 链 ， 直 到 链 中 的 某 个 解决 方案 能 够 处 
理 该 请 求 。 

通过 常量 相关 的 方法 ， 我 们 可 以 很 容易 地 实现 一 个 简单 的 职责 链 。 我 们 以 一 个 邮局 的 模型 
为 例 。 邮 局 需要 以 尽 可 能 通用 的 方式 来 处 理 每 一 封 邮件 ， 并 且 要 不 断 尝试 处 理 邮件 ， 直 到 该 邮 
件 最 终 被 确定 为 一 封 死 信 。 其 中 的 每 一 次 尝试 可 以 看 作为 一 个 策略 (也 是 一 个 设计 模式 )， 而 完 
整 的 处 理 方式 列表 就 是 一 个 职责 链 。 

我 们 先 来 描述 一 下 邮件 。 邮 件 的 每 个 关键 特征 都 可 以 用 enum 来 表示 。 程 序 将 随机 地 生成 
Mail 对 象 ， 如 果 要 减 小 一 封 邮 件 的 GeneralDelivery 为 YES 的 概率 ， 那 最 简单 的 方法 就 是 多 创建 
几 个 不 是 YES 的 enum 实 例 ， 所 以 enum 的 定义 看 起 来 有 点 古怪 。 

我 们 看 到 Mail 中 有 一 个 randomMail0) 方 法 ， 它 负责 随机 地 创建 用 于 测试 的 邮件 。 而 
generator() 方 法 生成 一 个 Iterable 对 象 ， 该 对 象 在 你 调用 next0 方 法 时 ， 在 其 内 部 使 用 random- 
Mail0 来 创建 Mail 对 象 。 这 样 的 结构 使 程序 员 可 以 通过 调用 Mail.generator0 方 法 ， 很 容易 地 构造 

出 一 个 foreach 循 环 : 


//: enumerated/PostOffice.java 
// Modeling a post office. 

import java.util.*; 

import net.mindview.util.*; 

import static net.mindview.util.Print.*; 


class Mail ( 
// The NO's lower the probability of random selection: 
enum GeneralDelivery {YES,NO1,NO2,NO3,NO4,NOS) 
enum Scannability (UNSCANNABLE, YES1,YES2,YES3,YES4) 
enum Readability (ILLEGIBLE, YES1,YES2,YES3, YES4) 
enum Address {INCORRECT OK1,OK2,0K3 ,OK4,,OKS ,OK6) 
enum ReturnAddress {MISSING OK1,0K2,0K3, OK4 ,0K5) 
GeneralDelivery generalDelivery; 
Scannability scannability; 
Readability readability; 
Address address; 
ReturnAddress returnAddress; 
static long counter = 0; 
long id = counter++; 
public String toString() { return “Mail " + id; } 
public String details() { 
return toString() + 
", General Delivery: ”+ generalDelivery + 
", Address Scanability: " + scannability + 
", Address Readability: " + readability + 
", Address Address: 
", Return address 








+ address + 
" + returnAddress; 





} 
// Generate test Mail: 
public static Mail randomMail() { 
Mail m = new Mail(); 
m.generalDelivery= Enums.random(GeneralDelivery.class): 
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m.scannability = Enums.random(Scannability.class); 
m.readability = Enums.random(Readability.class); 
m.address = Enums.random(Address.class); 
m.returnAddress = Enums.random(ReturnAddress.class); 
return m; 
} 
public static Iterable<Mail> generator(final int count) { 
return new Iterable<Mail>() { 
int n = count; 
public Iterator<Mail> iterator() { 
return new Iterator<Mail>() { 
public boolean hasNext() { return n-- > @; } 
Public Mail next() { return randomMail(): } 
public void remove() { // Not implemented 
throw new UnsupportedOperat ionException(); 











public class Postoffice { 
enum MailHandler { 
GENERAL_DELIVERY { 
boolean handle(Mail m) { 
switch(m.generalDelivery) { 
case YES: 
print("Using general delivery for " + m); 
return true; 
default: return false; 
+ 
} 
}, 


MACHINE_SCAN { 
boolean handle(Mail m) { 
switch(m.scannability) { 
case UNSCANNABLE: return false; 
default: 
switch(m.address) { 
case INCORRECT: return false; 
default: 
print("Delivering "+ m + * automatically"); 
return true; 


} 
} 
}, 
VISUAL_INSPECTION { 
boolean handle(Mail m) { 
switch(m.readability) { 
case ILLEGIBLE: return false; 





default: 
switch(m.address) { 
case INCORRECT: return false; 1038) 
default: 


print("Delivering " + m+ " normally"); 
return true; 


} 
} 


}, 
RETURN_TO_SENDER { 
boolean handle(Mail m) { 
switch(m.returnAddress) { 
case MISSING: return false; 
default: 











1039} 
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print("Returning * +m + " to sender"); 
return true; 
} 
$ 
Xi 
abstract boolean handle(Mail m); 
} 
static void handle(Mail m) { 
for(MailHandler handler : Ma{lHandler.values()) 
if (handler .handle(m)) 
return; 
print(m +" is a dead letter"); 
} 
public static void main(String{] args) { 
for(Mail mail : Mail.generator(1@)) { 
print (mail.details()); 
handle(mail) ; 
print("eesee" 
$ 
} 
} /* Output: 
Nail @, General Delivery: NO2, Address Scanability: 
UNSCANNABLE, Address Readability: YES3, Address Address: 
OK1, Return address: OK1 
Delivering Mail 9 normally 


Nail 1, General Delivery: NOS, Address Scanability: YES3， 
Address Readability: ILLEGIBLE, Address Address: 0KS, 
Return address: OK1 

Delivering Mail 1 automatically 

Mail 2, General Delivery: YES, Address Scanability: YES3, 
Address Readability: YES1, Address Address: OK1, Return 
address: OKS 

Using general delivery for Mail 2 

Mail 3, General Delivery: NO4, Address Scanability: YES3, 
Address Readability: YES1, Address Address: INCORRECT, 
Return address: OK4 

Returning Mail 3 to sender 

ee 

Mail 4, General Delivery: NO4, Address Scanability: 
UNSCANNABLE, Address Readability: YES1, Address Address: 
INCORRECT, Return address: OK2 

Returning Mail 4 to sender 

Mail 5, General Delivery: NO3, Address Scanability: YES1, 
Address Readability: ILLEGIBLE, Address Address: OK4, 
Return address: OK2 

Delivering Mail 5 automatically 

serae 

Mail 6, General Delivery: YES, Address Scanability: YES4, 
Address Readability: ILLEGIBLE, Address Address: OK4, 
Return address: OK4 

Using general delivery for Mail 6 

Nail 7, General Delivery: YES. Address Scanability: YES3, 
Address Readability: YES4, Address Address: OK2, Return 
address: MISSING 

Using general delivery for Mail 7 

stone 

Mail 8, General Delivery: NO3, Address Scanability: YES1, 
Address Readability: YES3, Address Address: INCORRECT, 
Return address: MISSING 

Mail 8 is a dead letter 


Mail 9, General Delivery: NO1, Address Scanability: 
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UNSCANNABLE, Address Readability: YES2, Address Address: 
OK1, Return address: OK4 
Delivering Mail 9 normally 


#41:~ 

职责 链 由 enum MailHandler 实 现 , 而 enum 定 义 的 次 序 决定 了 各 个 解决 策略 在 应 用 时 的 次 序 。 [Toad 
对 每 一 封 邮件 ， 都 要 按 此 顺序 尝试 每 个 解决 策略 ， 直 到 其 中 一 个 能 够 成 功 地 处 理 该 邮件 ， 如 果 
所 有 的 策略 都 失败 了 ， 那 么 该 邮件 将 被 判定 为 一 封 死 信 。 

练习 8: (6) 修改 PostOfficejava， 使 其 能 够 转发 邮件 。 

练习 9: (5) 修改 class PostOffice， 使 其 能 够 使 用 EnumMap。 

作业 ? : 专用 程序 设计 语言 ， 例 如 Prolog， 使 用 反 向 链 来 解决 类 似 的 问题 。 试 用 PostOffice 
java 做 一 个 例子 ， 研 究 一 下 这 些 语言 ， 用 其 编写 一 个 扩展 性 更 好 的 程序 ， 使 程序 员 可 以 很 容易 
地 向 系统 中 添加 新 的 “规则 ”。 
19.10.2 使 用 enum 的 状态 机 

枚 举 类 型 非常 适合 用 来 创建 状态 机 。 一 个 状态 机 可 以 具有 有 限 个 特定 的 状态 ， 它 通常 根据 
输入 ， 从 一 个 状态 转移 到 下 一 个 状态 ， 不 过 也 可 能 存在 办 时 状态 (transient states) ， 而 一 旦 任务 
执行 结束 ， 状 态 机 就 会 立刻 离开 瞬时 状态 。 

每 个 状态 都 具有 某 些 可 接受 的 输入 ， 不 同 的 输入 会 使 状态 机 从 当前 状态 转移 到 不 同 的 新 状 
态 。 由 于 enum 对 其 实例 有 严格 限制 ， 非 常 适合 用 来 表现 不 同 的 状态 和 输入 。 一 般 而 言 ， 每 个 状 
态 都 具有 一 些 相关 的 输出 。 

自动 售 货 机 是 一 个 很 好 的 状态 机 的 例子 。 首 先 ， 我们 用 一 个 enum 定 义 各 种 输入 : 


//: enumerated/Input.java 
package enumerated; 
import java.util.*; 














public enum Input ( 
NICKEL (5), DIME(1), QUARTER(25), DOLLAR(109), 
TOOTHPASTE(200), CHIPS(75), SODA(100), SOAP(56), 
ABORT_TRANSACTION { 
public int amount() { // Disallow 
throw new RuntimeException("ABORT. amount ()") ; 


3 


}, 
STOP { // This must be the last instance. 
public int amount() { // Disallow 
throw new RuntimeException("SHUT_DOWN. amount ()"); 
} 





目 


}; 
int value; // In cents 
Input(int value) { this.value = value; } 
Input() {} i 
int amount() { return value; }; // In cents 
static Random rand = new Random(47) ; 
public static Input randomSelection() { 
// Don't include STOP: 
return values()[rand.nextInt(values().length - 1)]; 


} 
dM hx 
注意 ， 除 了 两 个 特殊 的 Input 实 例 之 外 ， 其 他 的 Input 都 有 相应 的 价格 ， 因 此 在 接口 中 定义 
了 amount0 方 法 。 然 而 ， 对 那 两 个 特殊 Input 实 例 而 言 ， 调 用 amount0 方 法 并 不 合适 ， 所 以 如 果 
程序 员 调用 它们 的 amount0 方 法 就 会 有 异常 抛 出 《在 接口 内 定义 了 一 个 方法 ， 然 后 在 你 调用 该 


| 日 作业， 我 建议 读者 将 其 作为 课程 大 作业 。 解 答 指南 中 不 包含 此 类 作业 的 解决 方案 。 
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方法 的 某 个 实现 时 就 会 抛 出 异常 )。 这 似乎 有 点 奇怪 ， 但 由 于 enum 的 限制 ， 我 们 不 得 不 采用 这 


种 方式 。 
VendingMachine 对 输入 的 第 一 个 反应 是 将 其 归 类 为 Category enum 中 的 某 一 个 enum 实 例 ， 
这 可 以 通过 switch 实 现 。 下 面 的 例子 演示 了 enum 是 如 何 使 代码 变 得 更 加 清晰 且 易于 管理 的 : 


//: enumerated/VendingHachine.java 
// {Args: VendingMachineInput. txt} 
package enumerated; 

import java.util.*; 
import net.mindview.util.*; 

import static enumerated. Input. *; 

import static net.mindview.util.Print.*; 





enum Category { 
MONEY (NICKEL, DIME, QUARTER, DOLLAR), 
TTEM_SELECTION(TOOTHPASTE, CHIPS, SODA, SOAP), 
QUIT_TRANSACTION(ABORT_TRANSACTION) , 
‘SHUT_DOWN (STOP) ; 
private Input[] values; 
040] Category(Input... types) { values = types; } 
private static EnumMap<Input ,Category> categories = 
new EnumMap<Input ,Category>(Input.class) ; 
static { 
for (Category c : Category.class.getEnumConstants()) 
for(Input type : c.values) 
categories.put (type, €); 














} 
public static Category categorize(Input input) { 
return categories.get (input) ; 
) 
} 


public class VendingMachine { 
private static State state = State. RESTING; 
private static int amount = 0; 
private static Input selection = null; 
enum StateDuration { TRANSIENT } // Tagging enum 
enum State { 
RESTING { 
void next(Input input) { 
switch(Category.categorize(input)) { 
case MONEY: 
amount += input.amount(); 
state = ADDING_MONEY; 
break; 
case SHUT_DOWN: 
state = TERMINAL; 
default: 
} 
} 
h, 
ADDING_HONEY { 
void next(Input input) { 
switch(Category.categorize(input)) { 
case MONEY: 
amount += input .amount() ; 
break; 
case ITEM_SELECTION: 
selection = input; 
if(amount < selection. amount ()) 
print("Insufficient money for " + selection); 
else state = DISPENSING; 
break; 
case QUIT_TRANSACTION: 
state = GIVING_CHANGE: 
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break; 
case SHUT_DOWN: 
state = TERMINAL; 
default: 
} 
} 
DISPENSING(StateDuration. TRANSIENT) { 
void next() { 
print("here is your ”+ selection); 
amount -= selection.amount(); 
state = GIVING_CHANGE; 
} 
}, 
GIVING_CHANGE(StateDuration. TRANSIENT) { 
void next() { 
if(amount > 0) { 
print("Your change: ”+ amount); 
amount = @; Ş 
} 
state = RESTING; 
i ) 
TERMINAL { void output() { print("Halted"); } }; 
private boolean isTransient = false; 
State() {} 
State(StateDuration trans) { isTransient = true; } 
void next(Input input) { 
throw new RuntimeException("Only call " + 
“next(Input input) for non-transient states"); 
} 


void next() { 
throw new Runt imeException("Only call next() for " + 
"StateDuration. TRANSIENT states"); 
} 
void output() { print(amount); } 
} 
static void run(Generator<Input> gen) { 
while(state != State. TERMINAL) { 
state.next (gen.next()); 
while(state. isTransient) 
state.next(); 
state.output(); 


} 
} 
public static void main(String[] args) { 
Generator<Input> gen = new RandomInputGenerator(); 
if(args.length == 1) 
gen = new FileInputGenerator (args {[0]) ; 
run (gen) ; 
$ 
} 


// For a basic sanity check: 
class RandomInputGenerator implements Generator<Input> { 
public Input next() { return Input.randomSelection(); } 


} 


// Create Inputs from a file of ';'-separated strings: 
class FileInputGenerator implements Generator<Input> { 
private Iterator<String> input; 
public FileInputGenerator (String fileName) { 
input = new TextFile(fileName, ";").iterator(); 
} 
Public Input next() { 
if (!input.hasNext()) 
return null; 
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return Enum.valueOf(Input.class, input.next().trim()); 


} 
} /* Output: 
25 


here is your CHIPS 
9 


100 

200 

here is your TOOTHPASTE 
9 


25 
35 
Your change: 35 


35 


Insufficient money for SODA 
75 

Your change: 75 

9 


Halted 
Whim 


由 于 用 switch 语 句 从 enum 实 例 中 进行 选择 是 最 常见 的 一 种 方式 (请 注意 ， 为 了 使 enum 在 
switch 语 句 中 的 使 用 变 得 简单 ， 我 们 是 需要 付出 其 他 代价 的 ) ， 所 以 ， 我 们 经 常 遇 到 这 样 的 问题 : 
将 多 个 enum 进 行 分 类 时 ， “我们 希望 在 什么 enum 中 使 用 switeh 语 句 ? ”我 们 通过 VendingMachine 
的 例子 来 研究 一 下 这 个 问题 。 对 于 每 一 个 State， 我 们 都 需要 在 输入 动作 的 基本 分 类 中 进行 查找 ; 
用 户 塞 入 钞票， 选择 了 某 个 货物 ， 操 作 被 取消 ， 以 及 机 器 停止 。 然 而 ， 在 这 些 基 本 分 类 之 下 ， 我 
们 又 可 以 塞 人 不 同类 型 的 钞票 ， 可 以 选择 不 同 的 货物 。Category enum 将 不 同类 型 的 Input 进行 分 
组 ， 因 而 ， 可 以 使 用 categorize0 方 法 为 switch 语 句 生成 恰当 的 Cateroy 实 例 。 并 且 ， 访 方法 使 用 的 
EnumMap 确 保 了 在 其 中 进行 查询 时 的 效率 与 安全 。 

如 果 读 者 仔细 研究 VendingMachine 类 ， 就 会 发 现 每 种 状态 的 不 同 之 处 ， 以 及 对 于 输入 的 不 
同 响应 ， 其 中 还 有 两 个 瞬时 状态 。 在 run( 方 法 中 ， 状 态 机 等 待 着 下 一 个 Input， 并 一 直 在 各 个 状 
态 中 移动 ， 直 到 它 不 再 处 于 瞬时 状态 。 

通过 两 种 不 同 的 Generator 对 象 ， 我 们 可 以 用 两 种 方式 来 测试 VendingMachine。 首 先是 
RandomInputGenerator， 它 会 不 停 地 生成 各 种 输入 ， 当 然 ， 除 了 SHUT_DOWN 之 外 。 通 过 长 
时 间 地 运行 RandomInputGenerator， 可 以 起 到 健全 测试 (sanity test) 的 作用 ， 能 够 确保 该 状 
态 机 不 会 进入 一 个 错误 状态 。 另 一 个 是 FileInputGenerator， 使 用 文件 以 文本 的 方式 来 描述 输 
入 ， 然 后 将 它们 转换 成 enum 实 例 ， 并 创建 对 应 的 Input 对 象 。 上 面 的 程序 使 用 的 正 是 如 下 的 文 
本 文件 : 


//:! enumerated/VendingMachineInput. txt 
QUARTER; QUARTER; QUARTER; CHIPS; 
DOLLAR: DOLLAR; TOOTHPASTE; 

QUARTER: DIME; ABORT_TRANSACTION; 
QUARTER; DINE; SODA: 

QUARTER; DIME; NICKEL; SODA; 
‘ABORT_TRANSACTION: 

STOP; 

Whim 
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这 种 设计 有 一 个 缺陷 ， 它 要 求 enum State 实 例 访问 的 VendingMachine 属 性 必须 声明 为 static， 
这 意味 着 ， 你 只 能 有 一 个 VendingMachine 实 例 。 不 过 如 果 我 们 思考 一 下 实际 的 〈 嵌 入 式 Java) 
应 用 ， 这 也 许 并 不 是 一 个 大 问题 ， 因 为 在 一 台 机 器 上 ， 我 们 可 能 只 有 一 个 应 用 程序 。 

练习 10: (7) 修改 class VendingMachine， 在 其 中 使 用 EnumMap， 使 其 同时 可 以 存在 多 个 
VendingMachine 实 例 。 

练习 11: (7) 如 果 是 一 个 真 的 自动 售 货 机 ， 我 们 希望 能 够 很 容易 地 添加 或 改变 售卖 货品 的 种 
类 ， 因 此 ， 用 enum 来 表现 Input 时 的 缺陷 使 其 并 不 实用 (我们 知道 enum 对 实例 有 着 特殊 的 限制 ， 
一 旦 声明 结束 就 不 能 有 任何 改动 )。 修 改 VendingMachine， 用 一 个 class 来 表现 售卖 的 货品 ， 并 基 
于 一 个 文本 文件 ， 使 用 ArrayList 来 初始 化 它们 (可 以 使 用 net.mindview.util.TextFile)。 

作业 。 : 设计 一 个 自动 贩卖 机 ， 使 其 具备 能 够 很 容易 地 应 用 于 所 有 国家 的 国际 化 的 能 力 。 


19.11 多 路 分 发 


当 你 要 处 理 多 种 交互 类 型 时 ， 程 序 可 能 会 变 得 相当 杂乱 。 举 例 来 说 ， 如 果 一 个 系统 要 分 析 
和 执行 数学 表达 式 。 我 们 可 能 会 声明 Numberplus(Number)、Numbermultiple(Number) 等 等 ， 
其 中 Number 是 各 种 数字 对 象 的 超 类 。 然 而 ， 当 你 声明 a.plus(b) 时 ， 你 并 不 知道 3 或 b 的 确切 类 型 ， 
那 你 如 何 能 让 它们 正确 地 交互 呢 ? 

你 可 能 从 未 思考 过 这 个 问题 的 答案 。Java 只 支持 单 路 分 发 。 也 就 是 说 ， 如 果 要 执行 的 操作 
包含 了 不 止 一 个 类 型 未 知 的 对 象 时 ， 那 么 Java 的 动态 绑 定 机 制 只 能 处 理 其 中 一 个 的 类 型 。 这 就 
无 法 解决 我 们 上 面 提 到 的 问题 。 所 以 ， 你 必须 自己 来 判定 其 他 的 类 型 ， 从 而 实现 自己 的 动态 绑 
定 行为 。 

解决 上 面 问题 的 办 法 就 是 多 路 分 发 (在 那个 例子 中 ， 只 有 两 个 分 发 ， 一 般 称 之 为 两 路 分 发 ) 。 
多 态 只 能 发 生 在 方法 调用 时 ， 所 以 ， 如 果 你 想 使 用 两 路 分 发 ， 那 么 就 必须 有 两 个 方法 调用 : 第 
一 个 方法 调用 决定 第 一 个 未 知 类 型 ， 第 二 个 方法 调用 决定 第 二 个 未 知 的 类 型 。 要 利用 多 路 分 发 ， 
程序 员 必须 为 每 一 个 类 型 提供 一 个 实际 的 方法 调用 ， 如 果 你 要 处 理 两 个 不 同 的 类 型 体系 ， 就 需 
要 为 每 个 类 型 体系 执行 一 个 方法 调用 。 一 般 而 言 ， 程 序 员 需 要 有 设 定好 的 某 种 配置 ， 以 便 一 个 
方法 调用 能 够 引出 更 多 的 方法 调用 ， 从 而 能 够 在 这 个 过 程 中 处 理 多 种 类 型 。 为 了 达到 这 种 效果 ， 
我 们 需要 与 多 个 方法 一 同 工 作 : 因为 每 个 分 发 都 需要 一 个 方法 调用 。 在 下 面 的 例子 中 (实现 了 
“石头 、 剪 刀 、 布 ”游戏 ， 也 称 为 RoShamBo) 对 应 的 方法 是 compete0 和 eval0， 二 者 都 是 同一 个 
类 型 的 成 员 ， 它 们 可 以 产生 三 种 Outcome 实 例 中 的 一 个 作为 结果 ® : 


//: enumerated/Outcome. java 
package enumerated; 
public enum Outcome { WIN. LOSE, DRAW } ///:~ 


//: enumerated/RoShamBol. java 

// Demonstration of multiple dispatching. 
package enumerated; 

import java.util.*; 

‘import static enumerated.Outcome.*; 


interface Item { 
Outcome compete (Item it); 
Outcome eval (Paper p); 
Outcome eval (Scissors s); 


O 作业 ， 我 建议 读者 将 其 作为 课程 大 作业 。 解 答 指南 中 不 包含 此 类 作业 的 解决 方案 。 
| O 不 记得 是 在 哪 位 作者 的 书 中 读 到 的 ， 总 之 这 个 例子 已 经 存在 很 多 年 了 。 你 可 以 在 www.MindView ,net 上 找到 它 
们 的 C++ 和 Java 的 版 本 (FE (Thinking in Pattems》 中 )。 
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Outcome eval(Rock r); 
t 


class Paper implements Item { 
Toas] public Outcome compete(Item it) { return it.eval (this); } 
public Outcome eval (Paper p) { return DRAW; } 
public Outcome eval (Scissors s) { return WIN; 
public Outcome eval (Rock r) { return LOSE; } 
public String toString() { return “Paper”; } 
F 


class Scissors implements Item { 
public Outcome compete(Item it) { return it.eval(this); } 
public Outcome eval (Paper p) ( return LOSE; } 
public Outcome eval (Scissors s) { return DRAW: } 
public Outcome eval (Rock r) { return WIN: } 
public String toString() { return "Scissors 
} 


class Rock implements Item { 
public Outcome compete(Item it) { return it.eval(this); } 
Public Outcome eval (Paper p) { return WIN; } 
Public Outcome eval(Scissors s) { return LOSE; } 
Public Outcome eval(Rock r) { return DRAW: } 
Public String toString() { return "Rock": } 
} 


public class RoShamBol { 
static final int SIZE = 20; 
private static Random rand = new Random(47) ; 
public static Item newItem() { 
switch(rand.nextInt(3)) { 
defaul 








} 











return new Scissors(); 
return new Paper(); 
: return new Rock(); 





Public static void match(Item a, Item b) { 
System.out.priatin( 
at" vs. "+ b+": "+ a.compete(b)); 
} 
Public static void main(String(} args) { 
for(int i = 0; i < SIZE; i++) 
match(newItem(), newItem()): 
} 





} /* Output: 
Rock vs. Rock: DRAW 
11049) Paper vs. Rock: WIN 











Paper vs. Rock: WIN 
Paper vs. Rock: WIN 
Scissors vs. Paper: WIN 
Scissors vs. Scissors: DRAW 
Scissors vs. Paper: WIN 
Rock vs. Paper: LOSE 
Paper vs. Paper: DRAW 
Rock vs. Paper: LOSE 
Paper vs. Scissors: LOSE 
Paper vs. Scissors: LOSE 
Rock vs. Scissors: WIN 
Rock vs. Paper: LOSE 
Paper vs. Rock: WIN 
Scissors vs. Paper: WIN 
Paper vs. Scissors: LOSE 
Paper vs. Scissors: LOSE 
Paper vs. Scissors: LOSE 
Paper vs. Scissors: LOSE 
Whim 
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Item 是 这 几 种 类 型 的 接口 ， 将 会 被 用 作 多 路 分 发 。RoShamBol.matchO0 有 两 个 Item 参 数 ， 
通过 调用 Item.compete( 方 法 开始 两 路 分 发 。 要 判定 a 的 类 型 ， 分 发 机 制 会 在 a 的 实际 类 型 的 
compete0 内 部 起 到 分 发 的 作用 。compete0 方 法 通过 调用 eval0 来 为 另 一 个 类 型 实现 第 二 次 分 法 。 
将 自身 (this) 作为 参数 调用 eval0， 能 够 调用 重 载 过 的 eval0 方 法 ， 这 能 够 保留 第 一 次 分 发 的 类 
型 信息 。 当 第 二 次 分 发 完成 时 ， 你 就 能 够 知道 两 个 Item 对 象 的 具体 类 型 了 。 

要 配置 好 多 路 分 发 需要 很 多 的 工序 ， 不 过 要 记 住 ， 它 的 好 处 在 于 方法 调用 时 的 优雅 的 语法 ， 
这 避免 了 在 一 个 方法 中 判定 多 个 对 象 的 类 型 的 丑陋 代码 ， 你 只 需 说 ,“ 嘿 ， 你 们 两 个 ， 我 不 在 乎 
你 们 是 什么 类 型 ， 请 你 们 自己 交流 ! ”不 过 ， 在 使 用 多 路 分 发 前 ， 请 先 明确 ， 这 种 优雅 的 代码 
对 你 确实 有 重要 的 意义 。 

19.11.1 使 用 enum 分 发 

直接 将 RoShamBol.java 翻 译 为 基于 enum 的 版 本 是 有 问题 的 ， 因 为 enum 实 例 不 是 类 型 ， 不 
能 将 enum 实 例 作为 参数 的 类 型 ， 所 以 无 法 重 载 eval(0 方 法 。 不 过 ， 还 有 很 多 方式 可 以 实现 多 路 [050 
分 发 ， 并 从 enum 中 获 益 。 

一 种 方式 是 使 用 构造 器 来 初始 化 每 个 enum 实 例 ， 并 以 “一 组 ”结果 作为 参数 。 这 二 者 放 在 
一 块 ， 形 成 了 类 似 查询 表 的 结构 ; 


//: enumerated/RoShamBo2.java 

// Switching one enum on another. 
package enumerated; 

import static enumerated. Outcome.*; 





public enum RoShamBo2 implements Competitor<RoShamBo2> { 
PAPER(DRAW, LOSE, WIN), 
SCISSORS(WIN, DRAW, LOSE), 
ROCK(LOSE, WIN, DRAW) ; 
private Outcome vPAPER, vSCISSORS, vROCK; 
RoShamBo2 (Outcome paper „Outcome scissors,Qutcome rock) { 
this.vPAPER = paper; 
this. vSCISSORS = scissors; 
this.vROCK = rock; 
$ 
public Outcome compete(RoShamBo2 it) { 
switch(it) { 
default: 
case PAPER: return vPAPER; 
case SCISSORS: return vSCISSORS; 
case ROCK: return vROCK; 
} 


F 
public static void main(String[] args) { 
RoShamBo. play (RoShamBo2.class, 20); 


} 
} /* Output: 
ROCK vs. ROCK: DRAW 
SCISSORS vs. ROCK: LOSE 
SCISSORS vs. ROCK: LOSE 
SCISSORS vs. ROCK: LOSE 
PAPER vs. SCISSORS: LOSE 
PAPER vs. PAPER: DRAW 
PAPER vs. SCISSORS: LOSE 
| ROCK vs. SCISSORS: WIN 
SCISSORS vs. SCISSORS: DRAW 
ROCK vs. SCISSORS: WIN 
i SCISSORS vs. PAPER: WIN 1051] 
SCISSORS vs. PAPER: WIN 
ROCK vs. PAPER: LOSE 
| ROCK vs. SCISSORS: WIN 
SCISSORS vs. ROCK: LOSE 























1052 
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` PAPER vs. SCISSORS: LOSE 

SCISSORS vs. PAPER: WIN 

SCISSORS vs. PAPER: WIN 

SCISSORS vs. PAPER: WIN 

SCISSORS vs. PAPER: WIN 

H~ 

在 compete() 方 法 中 ， 一 旦 两 种 类 型 都 被 确定 了 ， 那 么 唯一 的 操作 就 是 返回 结果 Outcome。 
然而 ， 你 可 能 还 需要 调用 其 他 的 方法 ，( 例 如 ) 甚至 是 调用 在 构造 器 中 指定 的 某 个 命令 对 象 上 的 
方法 。 

RoShamBo2.java 比 之 前 的 例子 短小 得 多 ， 而 且 更 直接 ， 更 易于 理解 。 注 意 ， 我 们 仍然 是 使 
用 两 路 分 发 来 判定 两 个 对 象 的 类 型 。 在 RoShamBol.java 中 ， 两 次 分 发 都 是 通过 实际 的 方法 调用 
实现 ， 而 在 这 个 例子 中 ， 只 有 第 一 次 分 发 是 实际 的 方法 调用 。 第 二 个 分 发 使 用 的 是 switch， 不 过 
这 样 做 是 安全 的 ， 因 为 enum 限 制 了 switch 语 句 的 选择 分 支 。 

在 代码 中 ，enum 被 单独 抽取 出 来 ， 因 此 它 可 以 应 用 在 其 他 例子 中 。 首 先 ，Competitor 接 口 
定义 了 一 种 类 型 ， 该 类 型 的 对 象 可 以 与 另 一 个 Competitor 相 竞争 : 

//: enumerated/Competitor.java 


// Switching one enum on another. 
package enumerated; 


public interface Competitor<T extends Competitor<T>> { 
Outcome compete(T competitor); 
} Mi~ 


然后 ， 我 们 定义 两 个 statie 方 法 (static 可 以 避免 显 式 地 指明 参数 类 型 ) 。 第 一 个 是 match( 方 
法 ， 它 会 为 一 个 Competitor 对 象 调用 compete( 方 法 ， 并 与 另 一 个 Competitor 对 象 作 比较 。 在 这 
个 例子 中 ， 我 们 看 到 ，match( 方 法 的 参数 需要 是 Competitor<T> 类 型 。 但 是 在 Play0 方 法 中 ， 类 
型 参数 必须 同时 是 Enum<T> 类 型 (因为 它 将 在 Enums.random0 中 使 用 和 Competitor<T> 类 型 
(因为 它 将 被 传递 给 match( 方 法 ) : 

//: enumerated/RoShamBo. java 

// Common tools for RoShamBo examples. 


package enumerated; 
import net.mindview.util.*; 


public class RoShamBo { 
public static <T extends Competitor<T>> 
void match(T a, T b) { 
System. out. printin( 
a+" vs. "+b+": "+ a.compete(b)); 
} 
public static <T extends Enum<T> & Competitor<T>> 
void play(Class<T> rsbClass, int size) { 
for(int i = 0; 1 < size; i++) 
match( 
Enums.random(rsbClass) ,Enums.random(rsbClass)); 


} 

} Mi~ 

Play0 方 法 没有 将 类 型 参数 T 作 为 返回 值 类 型 ， 因 此 ， 似 乎 我 们 应 该 在 Class<T> 中 使 用 通 配 
符 来 代替 上 面 的 参数 声明 。 然 而 ， 通 配 符 不 能 扩展 多 个 基 类 ， 所 以 我 们 必须 采用 以 上 的 表达 式 。 
19.11.2 使 用 常量 相关 的 方法 

常量 相关 的 方法 允许 我 们 为 每 个 enum 实 例 提 供 方法 的 不 同 实现 ， 这 使 得 常量 相关 的 方法 似 
平 是 实现 多 路 分 发 的 完美 解决 方案 。 不 过 ， 通 过 这 种 方式 ，enum 实 例 虽然 可 以 具有 不 同 的 行为 ， 
但 它们 仍然 不 是 类 型 ， 不 能 将 其 作为 方法 签名 中 的 参数 类 型 来 使 用 。 最 好 的 办 法 是 将 enum 用 在 





| 
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Switch 语句 中 ， 见 下 例 : 


//: enumerated/RoShamBo3. java 
// Using constant-specific methods. 
package enumerated; 

import static enumerated.Qutcome.*; 


public enum RoShamBo3 implements Competitor<RoShamBo3> { 
PAPER { 
public Outcome compete (RoShamBo3 it) { 
Switch(it) { 
default: // To placate the compiler 
case PAPER: return DRAW; 
case SCISSORS: return LOSE; 1053) 
case ROCK: return WIN; 
} 
} 
}, 
SCISSORS { 
public Outcome compete(RoShamBo3 it) { 
switch(it) { 
default: 
case PAPER: return WIN; 
case SCISSORS: return DRAW 
case ROCK: return LOSE; 
} 
} 
}, 
ROCK { 
public Outcome compete(RoShamBo3 it) { 
switch(it) { 
default: 
case PAPER: return LOSE; 
case SCISSORS: return WIN; 
case ROCK: return DRAW; 
} 
H 
}; 
public abstract Outcome compete(RoShamBo3 it); 
public static void main(String{} args) { 
RoShamBo. play (RoShamBo3.class, 20); 


} 
} /* Same output as RoShamBo2.java *///:~ 


虽然 这 种 方式 可 以 工作 ， 但 是 却 不 甚 合理 ， 如 果 采 用 RoShamBo2.java 的 解决 方案 ， 那 么 在 
添加 一 个 新 的 类 型 时 ， 只 需 更 少 的 代码 ， 而 且 也 更 直接 。 
然而 ，RoShamBo3java 还 可 以 压缩 简化 一 下 : 


//: enumerated/RoShamBos. java 
package enumerated; 

















public enum RoShamBo4 implements Competitor<RoShamBo4> { 
ROCK { 
public Outcome compete(RoShamBo4 opponent) { 
return compete(SCISSORS, opponent) ; 
} 
}, 1054} 
SCISSORS { 
public Outcome compete(RoShamBo4 opponent) { 
return compete(PAPER, opponent); 
| } 
}, 
PAPER { 
public Outcome compete(RoShamBo4 opponent) { 
return compete(ROCK, opponent); 
} 
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he 
Outcome compete (RoShamBos loser, RoShamBo4 opponent) { 
return ((opponent == this) ? Outcome.DRAW 
: (opponent == loser) ? Outcome.WIN 
: Outcome. LOSE)); 
} 
public static void main(String[] args) { 
RoShamBo. play (RoShamBo4.class, 20); 


) }+ sane output as RoShamBo2. java *///:~ 

其 中 ， 具 有 两 个 参数 的 compete( 方 法 执行 第 二 个 分 发 ， 该 方法 执行 一 系列 的 比较 ， 其 行为 
类 似 switch 语 句 。 这 个 版 本 的 程序 更 简短 ， 不 过 却 比 较 难 理解 。 对 于 一 个 大 型 系统 而 言 ， 难 以 理 
解 的 代码 将 导致 整个 系统 不 够 健壮 。 
19.11.3 使 用 EnumMap 分 发 

使 用 EnumMap 能 够 实现 “真正 的 ”两 路 分 发 。EnumMap 是 为 enum 专 门 设计 的 一 种 性 能 非 
常 好 的 特殊 Map。 由 于 我 们 的 目的 是 摸索 出 两 种 未 知 的 类 型 ， 所 以 可 以 用 一 个 EnumMap 的 
EnumMap 来 实现 两 路 分 发 ; 


//: enumerated/RoShamBo5. java 
// Multiple dispatching using an EnumMap of EnumMaps. 
package enumerated; 

import java.util.*; 

import static enumerated. Outcome. *; 


enum RoShamBoS implements Competitor<RoShamBoS> { 
PAPER, SCISSORS, ROCK; 
static EnumMap<RoShamBoS , EnumMap<RoShamBoS , Outcome>> 
~ table = new EnumMap<RoShamBoS, 
EnumMap<RoShamBoS , Outcome>>(RoShamBoS .class) ; 
static ( 
for(RoShamBoS it : RoShamBoS.values()) 
table.put(it, 
new EnumMap<RoShamBo5 , Outcome>(RoShamBoS.class)); 
initRow(PAPER, DRAW, LOSE, WIN); 
initRow(SCISSORS, WIN, DRAW, LOSE); 
initRow(ROCK, LOSE, WIN, DRAW); 


} 
static void initRow(RoShamBoS it, 
Outcome vPAPER, Outcome vSCISSORS, Outcome vROCK) { 
EnumMap<RoShamBoS ,Outcome> row = 
RoShamBoS. table.get (it): 
row. put (RoShamBoS.PAPER, vPAPER); 
row. put (RoShamBoS . SCISSORS, vSCISSORS) ; 
row. put (RoShamBoS .ROCK, vROCK) ; 


} 

Public Outcome compete (RoShamBoS it) { 
return table.get(this).get (it); 

} 

public static void main(String[] args) { 
RoShamBo. play (RoShamBoS.class, 20); 


} 
} /* Same output as RoShamBo2.java *///:~ 


该 程序 在 一 个 static 子 名 中 初始 化 EnumMap 对 象 ， 具 体 见 表格 似 的 initRow0 方 法 调用 。 请 
注意 compete0 方 法 ， 您 可 以 看 到 ， 在 一 行 语句 中 发 生 了 两 次 分 发 。 
19.11.4 使 用 二 维 数组 

我 们 还 可 以 进一步 简化 实现 两 路 分 发 的 解决 方案 。 我 们 注意 到 ， 每 个 enum 实 例 都 有 一 个 固 
定 的 值 ( 基 于 其 声明 的 次 序 )， 并 且 可 以 通过 ordinal0 方 法 取得 该 值 。 因 此 我 们 可 以 使 用 二 维 数 
组 ， 将 竞争 者 映射 到 竞争 结果 。 采 用 这 种 方式 能 够 获得 最 简洁 、 最 直接 的 解决 方案 (很 可 能 也 
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是 最 快速 的 ， 虽 然 我 们 知道 EnumMap 内 部 其 实 也 是 使 用 数组 实现 的 ) 。 


//: enumerated/RoShamBo6. java 

// Enums using "tables" instead of multiple dispatch. 
package enumerated; 

import static enumerated.Qutcome.*; 


enum RoShamBo6 implements Competitor<RoShamBo6> { 
PAPER, SCISSORS, ROCK; 
private static Outcome[][] table = { 


{ DRAW, LOSE, WIN }, // PAPER 
{ WIN, DRAW, LOSE }, // SCISSORS 
{ LOSE, WIN, DRAW }, // ROCK 


fi 
public Outcome compete (RoShamBo6 other) { 
return table[this.ordinal()] [other.ordinal()}; 


} 
public static void main(String[] args) { 
RoShamBo. play(RoShamBo6.class, 20); 


$ 
) Mi~ 


table 与 前 一 个 例子 中 initRow( 方 法 的 调用 次 序 完全 相同 。 

与 前 面 一 个 例子 相 比 ， 这 个 程序 代码 虽然 简短 ， 但 表达 能 力 却 更 强 ， 部 分 原因 是 其 代码 更 
易于 理解 与 修改 ， 而 且 也 更 直接 。 不 过 ， 由 于 它 使 用 的 是 数组 ， 所 以 这 种 方式 不 太 “ 安 全 "。 如 
果 使 用 一 个 大 型 数组 ， 可 能 会 不 小 心 使 用 了 错误 的 尺寸 ， 而 且 ， 如 果 你 的 测试 不 能 覆盖 所 有 的 
可 能 性 ， 有 些 错误 可 能 会 从 你 眼前 溜 过 。 

事实 上 ， 以 上 所 有 的 解决 方案 只 是 各 种 不 同类 型 的 表 罢 了 。 不 过 ， 分 析 各 种 表 的 表现 形式 ， 
找 出 最 适合 的 那 一 种 ， 还 是 很 有 价值 的 。 注 意 ， 虽 然 上 例 是 最 简洁 的 一 种 解决 方案 ， 但 它 也 是 
相当 僵硬 的 方案 ， 因 为 它 只 能 针对 给 定 的 常量 输入 产生 常量 输出 。 然 而 ， 也 没有 什么 特别 的 理 
由 阻止 你 用 table 来 生成 功能 对 象 。 对 于 某 类 问题 而 言 ,“ 表 驱动 式 编码 ”的 概念 具有 非常 强大 的 
功能 。 


19.12 总 结 


虽然 枚 举 类 型 本 身 并 不 是 特别 复杂 ， 但 我 还 是 将 本 章 安排 在 全 书 比较 靠 后 的 位 置 ， 这 是 因 
为 ,程序 员 可 以 将 enum 与 Java 语 言 的 其 他 功能 结合 使 用 ， 例 如 多 态 、 泛 型 和 反射 

虽然 Java 中 的 枚 举 比 C 或 C++ 中 的 enum 更 成 熟 ， 但 它 仍然 是 一 个 “小 ”功能 ，Java 没 有 它 也 
已 经 (虽然 有 点 笨拙 ) 存在 很 多 年 了 。 而 本 章 正好 说 明了 一 个 “小 ”功能 所 能 带 来 的 价值 。 有 
时 恰恰 因为 它 ， 你 才能 够 优雅 而 干净 地 解决 问题 。 正 如 我 在 本 书 中 一 再 强调 的 那样 ， 优 雅 与 清 
晰 很 重要 ， 正 是 它们 区 别 了 成 功 的 解决 方案 与 失败 的 解决 方案 。 而 失败 的 解决 方案 就 是 因为 其 
他 人 无 法 理解 它 。 

关于 清晰 的 话题 ，Java 1.0 对 术语 enumeration 的 选择 正 是 一 个 不 幸 的 反例 。 对 于 一 个 专门 用 
于 从 序列 中 选择 每 一 个 元 素 的 对 象 而 言 ，Java 竞 然 没 有 使 用 更 通用 、 更 普遍 接受 的 术语 iterator 来 
表示 它 (参见 集合 )。 有 些 语言 甚至 将 枚 举 的 数据 类 型 称 为 enumerators! Java 修 正 了 这 个 错误 ， 
但 是 Enumeration 接 口 已 经 无 法 轻易 地 抹 去 了 ,因此 它 将 一 直 存在 于 旧 的 (甚至 有 些 新 的 ) 代码 、 
类 库 以 及 文档 中 。 

所 选 习题 的 答案 都 可 以 在 名 为 The Thinking in Java Annotated Solution Guide 的 电子 文档 中 找 
到 ， 读 者 可 以 从 www.MindView.net 购 买 此 文档 。 
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第 20 章 注 解 


注解 (也 被 称 为 元 数据 ) 为 我 们 在 代码 中 添加 信息 提供 了 一 种 形式 化 的 方法 ， 使 我 们 可 以 
在 稍 后 某 个 时 刻 非常 方便 地 使 用 这 些 数据 ? 。 

注解 在 一 定 程度 上 是 在 把 元 数据 与 源 代码 文件 结合 在 一 起 ， 而 不 是 保存 在 外 部 文档 中 这 一 
大 的 趋势 之 下 所 催生 的 。 同 时 ， 注 解 也 是 对 来 自 像 C# 之 类 的 其 他 语言 对 Java 造 成 的 语言 特性 压 
力 所 做 出 的 一 种 回应 。 

注解 是 众多 引入 到 Java SE5 中 的 重要 的 语言 变化 之 一 。 它 们 可 以 提供 用 来 完整 地 描述 程序 所 
需 的 信息 ， 而 这 些 信息 是 无 法 用 Java 来 表达 的 。 因 此 ， 注 解 使 得 我 们 能 够 以 将 由 编译 器 来 测试 
和 验证 的 格式 ， 存 储 有 关 程序 的 额外 信息 。 注 解 可 以 用 来 生成 描述 符 文件 ， 甚 至 或 是 新 的 类 定 
义 ， 并 且 有 助 于 减轻 编写 “样板 ”代码 的 负担 。 通 过 使 用 注解 ， 我 们 可 以 将 这 些 元 数据 保存 在 
Java 源 代码 中 ， 并 利用 annotation API 为 自己 的 注解 构造 处 理工 具 ， 同 时 ， 注 解 的 优点 还 包括 ， 
更 加 干净 易 读 的 代码 以 及 编译 期 类 型 检查 等 。 虽 然 Java SE5 预 先 定义 了 一 些 元 数据 , 但 一 般 来 说 ， 
主要 还 是 需要 程序 员 自己 添加 新 的 注解 ， 并 且 按 自己 的 方式 使 用 它们 。 

注解 的 语法 比较 简单 ， 除 了 @ 符 号 的 使 用 之 外 ， 它 基本 与 Java 固 有 的 语法 一 致 。Java SESH 
置 了 三 种 ， 定 义 在 javaang 中 的 注解 : 

+ @Override， 表 示 当 前 的 方法 定义 将 种 盖 超 类 中 的 方法 。 如 果 你 不 小 心 拼写 错误 ,或 者 方 

法 签名 对 不 上 被 覆盖 的 方法 ， 编 译 器 就 会 发 出 错误 提示 。 

，@Deprecated， 如 果 程序 员 使 用 了 注解 为 它 的 元 素 ， 那 么 编译 器 会 发 出 警告 信息 。 

*@SuppressWarnings， 关 闭 不 当 的 编译 器 警告 信息 。 在 Java SE5 之 前 的 版 本 中 ， 也 可 以 使 

用 该 注解 ， 不 过 会 被 忽略 不 起 作用 。 

Java 还 另外 提供 了 四 种 注解 ， 专 门 负责 新 注解 的 创建 。 稍 后 我 们 将 学 习 它 们 。 

每 当 你 创建 描述 符 性 质 的 类 或 接口 时 ， 一 旦 其 中 包含 了 重复 性 的 工作 ， 那 就 可 以 考虑 使 用 
注解 来 简化 与 自动 化 该 过 程 。 例 如 在 Enterprise JavaBean (EJB) 中 存在 许多 额外 的 工作 ， 
EJB3.0 就 是 使 用 注解 消除 了 它们 。 

注解 的 出 现 ， 可 以 玖 代 某 些 现存 的 系统 。 例 如 XDoclet (参见 http://MindView.net/Books/ 
BetterJava 的 附录 )， 它 是 一 个 独立 的 文档 化 工具 ， 专 门 设计 用 来 生成 类 似 注解 一 样 的 文档 。 与 之 
相 比 ， 注 解 是 真正 的 语言 级 的 概念 ， 一 旦 构造 出 来 ， 就 享有 编译 期 的 类 型 检查 保护 。 注 解 
(annotation) 是 在 实际 的 源 代码 级 别 保存 所 有 的 信息 ， 而 不 是 某 种 注释 性 的 文字 (comment), x 
使 得 代码 更 整洁 ， 且 便于 维护 。 通 过 使 用 扩展 的 annotation API， 或 外 部 的 字 节 码 工具 类 库 ( 稍 
后 你 将 会 看 到 )， 程 序 员 拥有 对 源 代码 以 及 字 节 码 强大 的 检查 与 操作 能 力 。 


20.1 基本 语法 
在 下 面 的 例子 中 ， 使 用 @Test 对 testExecute() 方 法 进行 注解 。 该 注解 本 身 并 不 做 任何 事情 ， 
© Jeremy Meyer 来 到 Crested Butte 花 了 两 周 的 时 间 和 我 一 起 撰写 本 章 ， 他 的 帮助 弥 足 珍贵 


O 这 无 疑 是 受到 C# 中 类 似 功能 的 启发 。C# 的 这 项 功能 源 自 一 个 关键 字 ， 而 不 是 注解 ， 并 且 是 编译 器 强制 要 求 的 。 
也 就 是 说 ， 当 C# 得 序 员 覆 姜 一 个 方法 时 ， 必 须 使 用 override 关 键 字 ， 而 在 Java 中 ，@Override 是 可 选择 的 。 
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但 是 编译 器 要 确保 在 其 构造 路 径 上 必须 有 @Test 注 解 的 定义 。 你 将 在 本 章 中 看 到 ， 程 序 员 可 以 创 
建 一 个 通过 反射 机 制 来 运行 testExecute0 方 法 的 工具 。 


//: annotations/Testable.java 
package annotations; 
‘import net.mindview.atunit.*: 1060] 
public class Testable { 
public void execute() { 
System.out.printin("Executing.."); 














》 
@Test void testExecute() {. execute(); } 
} Mi~ 


被 注解 的 方法 与 其 他 的 方法 没有 区 别 。 在 这 个 例子 中 ， 注 解 @Test 可 以 与 任何 修饰 符 共同 作 
用 于 方法 ， 例 如 pablic、static 或 void。 从 语法 的 角度 来 看 ， 注 解 的 使 用 方式 几乎 与 修饰 符 的 使 
用 一 模 一 样 。 
20.1.1 定义 注解 

下 面 就 是 前 例 中 用 到 的 注解 @Test 的 定义 。 可 以 看 到 ， 注 解 的 定义 看 起 来 很 像 接口 的 定义 。 
事实 上 ， 与 其 他 任何 Java 接 口 一 样 ， 注 解 也 将 会 编译 成 class 文 件 。 


//: net/mindview/atunit/Test.java 
// The @Test tag. 

package net.mindview.atunit; 
import java. lang. annotation.*; 


@Target (ElementType. METHOD) 
@Retent ion(Retent ionPol icy .RUNTIME) 
public @interface Test {} ///:~ 


除了 @ 符 号 以 外 ，@Test 的 定义 很 像 一 个 空 的 接口 。 定 义 注解 时 ， 会 需要 一 些 元 注解 
(meta-annotation), ， 如 @Target 和 @Retention。@Targef 用 米 定 义 你 的 注解 将 应 用 于 什么 地 方 
(例如 是 一 个 方法 或 者 一 个 域 )。@Rectetion 用 来 定义 该 注解 在 哪 一 个 级 别 可 用 ， 在 源 代码 中 
(SOURCE)、 类 文件 中 (CLASS) 或 者 运行 时 (RUNTIME)。 

在 注解 中 ， 一 般 都 会 包含 一 些 元 素 以 表示 某 些 值 。 当 分 析 处 理 注解 时 ， 程 序 或 工具 可 以 利 
用 这 些 值 。 注 解 的 元 素 看 起 来 就 像 接口 的 方法 ， 唯 一 的 区 别 是 你 可 以 为 其 指定 默认 值 。 

没有 元 素 的 注解 称 为 标记 注解 (marker annotation)， 例 如 上 例 中 的 @Test。 

下 面 是 一 个 简单 的 注解 ， 我 们 可 以 用 它 来 跟踪 一 个 项 目 中 的 用 例 。 如 果 一 个 方法 或 一 组 方 
法 实现 了 某 个 用 例 的 需求 ， 那 么 程序 员 可 以 为 此 方法 加 上 该 注解 。 于 是 ， 项 目 经 理 通过 计算 已 
经 实现 的 用 例 ， 就 可 以 很 好 地 掌控 项 目的 进展 。 而 如 果 要 更 新 或 修改 系统 的 业务 逻辑 ， 则 维护 
该 项 目的 开发 人 员 也 可 以 很 容易 地 在 代码 中 找到 对 应 的 用 例 。 


//: annotations/UseCase. java 
import java. lang. annotation. *; 


@Target (ElementType. METHOD) 
@Retent ion(Retent ionPol icy.RUNTIME) 
public @interface UseCase { 

public int id(); 

public String description() default "no description"; 
} Mi~ 


注意 ，id 和 description 类 似 方法 定义 。 由 于 编译 器 会 对 id 进行 类 型 检查 ， 因 此 将 用 例文 档 的 
追踪 数据 库 与 源 代码 相关 联 是 可 靠 的 。description 元 素 有 一 个 default 值 ， 如 果 在 注解 某 个 方法 
时 没有 给 出 deseription 的 值 ， 则 该 注解 的 处 理 器 就 会 使 用 此 元 素 的 默认 值 。 

在 下 面 的 类 中 ， 有 三 个 方法 被 注解 为 用 例 : 
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//: annotations/PasswordUtils.java 
import java.util.*; 


public class PasswordUtils { 
@UseCase(id = 47, description = 
"Passwords must contain at least one numeric”) 
public boolean validatePassword(String password) { 
return (password.matches("\\w*\\d\\w*")) ; 


} 

@UseCase(id = 48) 

public String encryptPassword(String password) { 

return new StringBuilder (password) .reverse().toString(): 


} 
@UseCase(id = 49, description = 
"New passwords can't equal previously used ones") 
public boolean checkForNewPassword( 
List<String> prevPasswords, String password) { 
return !prevPasswords.contains (password) ; 


} 
} Mi~ 


注解 的 元 素 在 使 用 时 表现 为 名 - 值 对 的 形式 ， 并 需要 置 于 @UseCase 声 明之 后 的 括号 内 。 在 
eneryptPassword() 方 法 的 注解 中 ， 并 没有 给 出 description 元 素 的 值 ， 因 此 ， 在 UseCase 的 注解 处 
理 器 分 析 处 理 这 个 类 时 会 使 用 该 元 素 的 默认 值 。 

你 应 该 能 够 想象 得 到 如 何 使 用 这 套 工具 来 “勾勒 ”出 将 要 建造 的 系统 ， 然 后 在 建造 的 过 程 
中 逐渐 实现 系统 的 各 项 功能 。 

20.1.2 元 注解 

Java 目 前 只 内 置 了 三 种 标准 注解 〈 前 面 介 绍 过 ) ， 以 及 四 种 元 注解 。 元 注解 专职 负责 注解 其 

他 的 注解 : 


@Target 表示 该 注解 可 以 用 干什么 地 方 。 可 能 的 ElementType 参 数 包括 : 
CONSTRUCTOR: 构造 器 的 声明 
FIELD; 域 声明 (包括 enum 实 例 ) 
LOCAL_VARIABLE, 局 部 变量 声明 
METHOD; 方法 声明 
PACKAGE, 包 声明 
PARAMETER， 参数 声明 
TYPE: 类 、 接 口 (包括 注解 类 型 ) 或 enum 疡 明 
@Retention 表示 需要 在 什么 级 别 保存 该 注解 信息 。 可 选 的 RetentionPolicy 参 数 包括 : 
SOURCE, 注解 将 被 编译 器 丢弃 。 
CLASS: 注解 在 class 文 件 中 可 用 ， 但 会 被 YM 丢弃 。 
RUNTIME: VM 将 在 运行 期 也 保留 注解 ， 因 此 可 以 通过 反射 机 制 读 到 注解 的 信息 。 
@Documented 将 此 注解 包含 在 Javadoc 中 。 


@Inherited 克 许 子 类 继承 父 类 中 的 注解 
大 多 数 时 候 ， 程 序 员 主 要 是 定义 自己 的 注解 ， 并 编写 自己 的 处 理 器 来 处 理 它们 。 
20.2 编写 注解 处 理 器 


如 果 没有 用 来 读 取 注 解 的 工具 ， 那 注解 也 不 会 比 注释 更 有 用 。 使 用 注解 的 过 程 中 ， 很 重要 
的 一 个 部 分 就 是 创建 与 使 用 注解 处 理 器 。Java SE5 扩 展 了 反射 机 制 的 API， 以 帮助 程序 员 构 造 这 
类 工具 。 同 时 ， 它 还 提供 了 一 个 外 部 工具 apt 帮 助 程序 员 解 析 带 有 注解 的 Java 源 代码 。 

下 面 是 一 个 非常 简单 的 注解 处 理 器 ， 我 们 将 用 它 来 读 取 PasswordUtils 类 ， 并 使 用 反射 机 制 
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查找 @UseCase 标 记 。 我 们 为 其 提供 了 一 组 id 值 ， 然 后 它 会 列 出 在 PasswordUtils 中 找到 的 用 例 ， 
以 及 缺失 的 用 例 。 


//: annotations/UseCaseTracker .java 
import java.lang.reflect.*; 
import java.util.*; 


public class UseCaseTracker { 
public static void 
trackUseCases(List<Integer> useCases, Class<?> cl) { 
for(Method m : cl.getDeclaredMethods()) { 
UseCase uc = m.getAnnotation(UseCase.class) ; 
if(uc != null) { 
System.out.printin("Found Use Case:" + uc.id() + 
" "+ uc.description()): 
useCases.remove(new Integer (uc.id())); 


} 
for(int 1 : useCases) { 
System.out.printIn("Warning: Missing use case-" + i); 

} 

} 

public static void main(String{] args) { 
List<Integer> useCases = new ArrayList<Integer>(); 
Collections. addAll(useCases, 47, 48, 49, 50); 
trackUseCases(useCases, PasswordUtils.class); 


} 
} /* Output: 
Found Use Case:47 Passwords must contain at least one 
numeric 
Found Use Case:48 no description 
Found Use Case:49 New passwords can't equal previously used 
ones 
Warning: Missing use case-50 
IIi~ 


这 个 程序 用 到 了 两 个 反射 的 方法 ， getDeclaredMethods0 和 getAnnotation0， 它 们 都 属于 
AnnotatedElement 接 口 (Class、Method 与 Field 等 类 都 实现 了 该 接口 )。getAnnoation0 方 法 返 
回 指定 类 型 的 注解 对 象 ， 在 这 里 就 是 UseCase。 如 果 被 注解 的 方法 上 没有 该 类 型 的 注解 ， 则 返回 
null 值 。 然 后 我 们 通过 调用 id0 和 description0 方 法 从 返回 的 UseCase 对 象 中 提取 元 素 的 值 。 其 中 ， 
encriptPassword() 方 法 在 注解 的 时 候 没 有 指定 description 的 值 ， 因 此 处 理 器 在 处 理 它 对 应 的 注解 
时 ， 通 过 description() 方 法 取得 的 是 默认 值 no description, 

20.2.1 注解 元 素 

标签 @UseCase 由 UseCasejava 定 义 ， 其 中 包含 int 元 素 id， 以 及 一 个 String 元 素 description 。 
注解 元 素 可 用 的 类 型 如 下 所 示 : 

* 所 有 基本 类 型 (int，foat，boolean 等 ) 

+ String 

+ Class 

enum 

* Annotation. 

“以 上 类 型 的 数组 

如 果 你 使 用 了 其 他 类 型 ， 那 编译 器 就 会 报错 。 注 意 ， 也 不 允许 使 用 任何 包装 类 型 ， 不 过 由 
于 自动 打包 的 存在 ， 这 算 不 是 什么 限制 。 注 解 也 可 以 作为 元 素 的 类 型 ， 也 就 是 说 注解 可 以 嵌 套 ， 
稍 后 你 会 看 到 ， 这 是 一 个 很 有 用 的 技巧 。 
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20.2.2 默认 值 限制 

编译 器 对 元 素 的 默认 值 有 些 过 分 挑 别 。 首 先 ， 元 素 不 能 有 不 确定 的 值 。 也 就 是 说 ,元 素 必 
须要 么 具有 默认 值 ， 要 么 在 使 用 注解 时 提供 元 素 的 值 。 

其 次 ， 对 于 非 基本 类 型 的 元 素 ， 无论 是 在 源 代码 中 声明 时 ， 或 是 在 注解 接口 中 定义 默认 值 
时 ， 都 不 能 以 null 作 为 其 值 。 这 个 约束 使 得 处 理 器 很 难 表现 一 个 元 素 的 存在 或 缺失 的 状态 ， 因 为 
在 每 个 注解 的 声明 中 ， 所 有 的 元 素 都 存在 ， 并 且 都 具有 相应 的 值 。 为 了 绕 开 这 个 约束 ， 我 们 只 
能 自己 定义 一 些 特殊 的 值 ， 例 如 空 字符 串 或 负数 ， 以 此 表示 某 个 元 素 不 存在 ; 


//: annotations/SimulatingNull. java 
import java.lang.annotation.*; 


@Target (ElementType METHOD) 
@Retention(RetentionPolicy. RUNTIME) 
public @interface SimulatingNull { 
public int id() default -1; 
public String description() default ""; 
} Mi~ 


在 定义 注解 的 时 候 ， 这 算得 上 是 一 个 习惯 用 法 。 
20.2.3 生成 外 部 文件 

有 些 framework 需 要 一 些 额外 的 信息 才能 与 你 的 源 代码 协同 工作 ， 而 这 种 情况 最 适合 注解 表 
现 其 价值 了 。 像 (EJB3 之 前 ) Enterprise JavaBean 这 样 的 技术 ， 每 一 个 Bean 都 需要 大 量 的 接口 和 
部 署 来 描述 文件 ， 而 这 些 都 属于 “样板 ”文件 。Web Service、 自 定义 标签 库 以 及 对 象 /关系 映射 
工具 (例如 Toplink 和 Hibernate) 等 ， 一 般 都 需要 XML 描述 文件 ， 而 这 些 描述 文件 脱离 于 源 代码 
之 外 。 因 此 ， 在 定义 了 Java 类 之 后 ， 程 序 员 还 必须 得 忍受 着 沉闷 ， 重 复 地 提供 某 些 信息 ， 例 如 
类 名 和 包 名 等 已 经 在 原始 的 类 文件 中 提供 了 的 信息 。 每 当 程序 员 使 用 外 部 的 描述 文件 时 ， 他 就 
拥有 了 同一 个 类 的 两 个 单独 的 信息 源 ， 这 经 常 导 致 代码 同步 问题 。 同 时 ， 它 也 要 求 为 项 目 工作 
的 程序 员 ， 必 须 同 时 知道 如 何 编写 Java 程 序 ， 以 及 如 何 编辑 描述 文件 。 

假设 你 希望 提供 一 些 基本 的 对 象 /关系 映射 功能 ， 能 够 自动 生成 数据 库 表 ， 用 以 存储 
JavaBean 对 象 。 你 可 以 选择 使 用 XML 描述 文件 ， 指 明 类 的 名 字 、 每 个 成 员 以 及 数据 库 映 射 的 相 
关 信 息 。 然 而 ， 如 果 使 用 注解 的 话 ， 你 可 以 将 所 有 信息 都 保存 在 JavaBean 源 文件 中 。 为 此 ， 我 
们 需要 一 些 新 的 注解 ， 用 以 定义 与 Bean 关 联 的 数据 库 表 的 名 字 ， 以 及 与 Bean 属 性 关联 的 列 的 名 
字 和 SQL 类 型 。 

以 下 是 一 个 注解 的 定义 ， 它 告诉 注解 处 理 器 ， 你 需要 为 我 生成 一 个 数据 库 表 ; 


//: annotations/database/DBTabte.java 
package annotations.database; 
import java. 1ang.annotation.*; 


@Target (ElementType.TYPE) // Applies to classes only 
@Retent jon (RetentionPol icy RUNTIME) 
public @interface DBTable { 
public String name() default *"; 
} Ii~ 


在 @Target 注 解 中 指定 的 每 一 个 ElementType 就 是 一 个 约束 ， 它 告诉 编译 器 ， 这 个 自 定义 的 
注解 只 能 应 用 于 该 类 型 。 程 序 员 可 以 只 指定 enum ElementType 中 的 某 一 个 值 ， 或 者 以 逗号 分 隔 
的 形式 指定 多 个 值 。 如 果 想 要 将 注解 应 用 于 所 有 的 ElementType， 那 么 可 以 省 去 @Target 元 注解 ， 
不 过 这 并 不 常见 。 

注意 ，@DBTable 有 一 个 name0 元 素 ， 该 注解 通过 这 个 元 素 为 处 理 器 创建 数据 库 表 提供 表 的 
名 字 。 
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接 下 来 是 为 修饰 JavaBean 域 准备 的 注解 : 


/1: annotations/database/Constraints. java 
package annotations.database; 
import java. lang. annotation. *; 


@Target (ElementType. FIELD) 
@Retent ion(Retent ionPol icy..RUNTIME) 
public @interface Constraints { $ 
boolean primaryKey() default false; 
boolean allowNull() default true; 
boolean unique() default false; 
Mh 
/1/: annotations/database/SQLstring. java 
package annotations .database; 
import java. lang. annotatton.*; 
@Target (ElementType. FIELD) 
@Retent ion (Retent ionPolicy .RUNTIME) 
public @interface SQLString { 
int value() default 0; 
String name() default ""; è 
Constraints ‘constraints() default @Constraints; 
} Mi~ 
//: annotations/database/SQLInteger. java 
package annotations .database; 
import java. lang. annotation. *; 


@Target(ElementType. FIELD) 2 
@Retention(Retent ionPolicy .RUNTIME) 
public @interface SQLInteger { 

String name() default ""; 

Constraints constraints() default @Constraints; 
ym 


注解 处 理 器 通过 @Constraints 注 解 提 取出 数据 库 表 的 元 数据 。 虽 然 对 于 数据 库 所 能 提供 的 
所 有 约束 而 言 ，@Constraints 注 解 只 表示 了 它 的 一 个 很 小 的 子 集 ， 不 过 它 所 要 表达 的 思想 已 经 
很 清楚 了 。primaryKey0、allowNull0 和 unique0 元 素 明 智 地 提供 了 默认 值 ， 从 而 在 大 多 数 情况 
下 ， 使 用 该 注解 的 程序 员 无 需 输 入 太 多 东西 。 

另外 两 个 @interface 定 义 的 是 SQL 类 型 。 如 果 希 望 这 个 framework 更 有 价值 的 话 ， 我 们 就 应 
该 为 每 种 SQL 类 型 都 定义 相应 的 注解 。 不 过 作为 示例 ， 两 个 类 型 足够 了 。 

这 些 SQL 类 型 具有 name0 元 素 和 constraints0 元 素 。 后 者 利用 了 人 网 套 注解 的 功能 ， 将 column 
类 型 的 数据 库 约束 信息 伐 入 其 中 。 注 意 constraints0 元 素 的 默认 值 是 @Constraints。 由 于 在 
@Constraints 注 解 类 型 之 后 ， 没 有 在 括号 中 指明 @Constraints 中 的 元 素 的 值 ， 因 此 ， 
constraints() 元 素 的 默认 值 实际 上 就 是 一 个 所 有 元 素 都 为 默认 值 的 @Constraits 注 解 。 如 果 要 令 
代入 的 @Constraints 注 解 中 的 unique0 元 素 为 true， 并 以 此 作为 constraints0 元 素 的 默认 值 ， 则 需 
要 如 下 定义 该 元 素 : 


/1/: annotations/database/Uniqueness.java 
// Sample of nested annotations 
package annotations.. database; 


public @interface Uniqueness { 
Constraints constraints() 
default @Constraints(unique=true) ; 
} ki~ 


下 面 是 一 个 简单 的 Bean 定 义 ， 我 们 在 其 中 应 用 了 以 上 这 些 注解: 


//: annotations/daťabase/Member: java 


[oo 
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package annotations. database; 


@DBTable(name = "MEMBER") 
public class Member { 
@SQLString(30) String firstName; 
@SQLString(5@) String lastName; 
@sQLInteger Integer age: 
@SQLString(value = 30, 
constraints = @Constraints(primaryKey = true)) 
String handle; 
static int memberCount; 
public String getHandle() { return handle; } 
public String getFirstName() { return firstName; } 
public String getLastName() { return lastName; } 
public String toString() { return handle: } 
public Integer getAge() { return age; } 
} Mi~ 


类 的 注解 @DBTable 给 定 了 值 MEMBER， 它 将 会 用 来 作为 表 的 名 字 。Bean 的 属性 firstName 
和 lastName， 都 被 注解 为 @SQLString 类 型 ， 并 且 其 元 素 值 分 别 为 30 和 50。 这 些 注解 有 两 个 有 趣 
的 地 方 : 第 一 ， 他 们 都 使 用 了 嵌入 的 @Constraints 注 解 的 默认 值 ， 第 二 ， 它 们 都 使 用 了 快捷 方 
式 。 何 谓 快捷 方式 呢 ， 如 果 程 序 员 的 注解 中 定义 了 名 为 value 的 元 素 ， 并 且 在 应 用 该 注解 的 时 候 ， 
如 果 该 元 素 是 唯一 需要 赋值 的 一 个 元 素 ， 那 么 此 时 无 需 使 用 名 - 值 对 的 这 种 语法 ， 而 只 需 在 括号 
内 给 出 value 元 素 所 需 的 值 即 可 。 这 可 以 应 用 于 任何 合法 类 型 的 元 素 。 当 然 了 ， 这 也 限制 了 程序 
员 必 须 将 此 元 素 命 名 为 value， 不 过 在 上 面 的 例子 中 ， 这 不 但 使 语义 更 清晰 ， 而 且 这 样 的 注解 语 
句 也 更 易于 理解 ; 

@SQLString(36) 

处 理 器 将 在 创建 表 的 时 候 使 用 该 值 设 置 SQL 列 的 大 小 。 

默认 值 的 语法 虽然 很 灵巧 ， 但 它 很 快 就 变 得 复杂 起 来 。 以 handle 域 的 注解 为 例 ， 这 是 一 个 
@SQLString 注 解 ， 同 时 该 域 将 成 为 表 的 主键 ， 因 此 在 做 入 的 @Constraints 注 解 中 ， 必 须 对 
PrimaryKey 元 素 进行 设 定 。 这 时 事情 就 变 得 麻烦 了 。 现 在 ， 你 不 得 不 使 用 很 长 的 名 - 值 对 形式 ， 
重新 写 出 元 素 名 和 @interface 的 名 字 。 与 此 同时 ， 由 于 有 特殊 命名 的 value 元 素 已 经 不 再 是 唯一 
需要 赋值 的 元 素 了 ， 所 以 你 也 不 能 再 使 用 快捷 方式 为 其 赋值 了 。 如 你 所 见 ， 最 终 的 结果 算 不 上 
清晰 易 懂 。 

变通 之 道 

可 以 使 用 多 种 不 同 的 方式 来 定义 自己 的 注解 ， 以 实现 上 例 中 的 功能 。 例 如 ， 你 可 以 使 用 一 
个 单一 的 注解 类 @TableColumn， 它 带 有 一 个 enum 元 素 ， 该 枚 举 类 定义 了 STRING、INTEGER 
以 及 FLOAT 等 枚 举 实例 。 这 就 消除 了 每 个 SQL 类 型 都 需要 一 个 @interface 定 义 的 负担 ， 不 过 也 使 
得 以 额外 的 信息 修饰 SQL 类 型 的 需求 变 得 不 可 能 ， 而 这 些 额外 的 信息 ， 例如 长 度 或 精度 等 ， 可 
能 是 非常 有 必要 的 需求 。 

我 们 也 可 以 使 用 String 元 素来 描述 实际 的 SQL 类 型 ， 比 如 VARCHAR(30) 或 INTEGER。 这 使 
得 程序 员 可 以 修饰 SQL 类 型 。 但 是 ， 它 同时 也 将 Java 类 型 到 SQL 类 型 的 映射 绑 在 了 一 起 ， 这 可 不 
是 一 个 好 的 设计 。 我 们 可 不 希望 更 换 数据 库 导致 代码 必须 修改 并 重新 编译 。 如 果 我 们 只 需 告诉 
注解 处 理 器 ， 我 们 正在 使 用 的 是 什么 “口味 ”的 SQL， 然 后 由 处 理 器 为 我 们 处 理 SQL 类 型 的 细 
节 ， 那 将 是 一 个 优雅 的 设计 。 

第 三 种 可 行 的 方案 是 同时 使 用 两 个 注解 类 型 来 注解 一 个 域 ， @Constraints 和 相应 的 SQL 类 
型 (例如 @SQLIntege)。 这 种 方式 可 能 会 使 代码 有 点 乱 ， 不 过 编译 器 允许 程序 员 对 一 个 目标 同 
时 使 用 多 个 注解 。 注 意 ， 使 用 多 个 注解 的 时 候 ， 同 一 个 注解 不 能 重复 使 用 。 
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20.2.4 注解 不 支持 继承 
不 能 使 用 关键 字 extends 来 继承 某 个 @interface。 这 真是 一 个 遗憾 。 如 果 可 以 定义 一 个 
@TableColumn 注 解 (参考 前 面 的 建议 )， 同 时 在 其 中 嵌 套 一 个 @SQLType 类 型 的 注解 ， 那 么 这 
将 成 为 一 个 优雅 的 设计 。 按 照 这 种 方式 ， 程序 员 可 以 继承 @SQLType， 从 而 创建 出 各 种 SQL 类 
| 型 , 例如 @SQLInteger 和 @SQLString 等 。 如 果 注 解 允 许 继承 的 话 ， 这 将 大 大 减少 打字 的 工作 量 ， 
| 并 且 使 语法 更 整洁 。 在 Java 未 来 的 版 本 中 ， 似 乎 没有 任何 关于 让 注解 支持 继承 的 提案 ， 所 以 ， 
在 当前 状况 下 ， 上 例 中 的 解决 方案 可 能 已 经 是 最 佳 方法 了 。 


20.2.5 实现 处 理 器 
i 下 面 是 一 个 注解 处 理 器 的 例子 ， 它 将 读 取 一 个 类 文件 ， 检 查 其 上 的 数据 库 注解 ， 并 生成 用 
来 创建 数据 库 的 SQL 命令 : 


//: annotations/database/TableCreator.java 
// Reflection-based annotation processor. 
// {Args: annotations database. Member} 
package annotations .databas: 

import java.lang.annotation.*; 

import java.lang.reflect.*; 
import java.util.*; 








public class TableCreator { 
public static void main(String{] args) throws Exception { 
if(args.length < 1) { 
System.out.printin("arguments: annotated classes"); 
System. exit (0); 


t 
for(String className : args) { 
Class<?> cl = Class. forName(className) ; 
DBTable dbTable = cl.getAnnotat ion(DBTable.class) ; 
if(dbTable == null) { 
System. out.printin( 
"No DBTable annotations in class ”+ className); 
continue; 
} 
String tableName = dbTable.name(); 
// If the name is empty, use the Class name: 
if(tableName.length() < 1) 
tableName = cl.getName().toUpperCase(); 
List<String> columnDefs = new ArrayList<String>(); 
for(Field field : cl.getDeclaredFields()) { 
String columnName = null; 
Annotation[] anns = field.getDeclaredAnnotations(); 
if(anns. length < 1) 
continue; // Not a db table column 
if(anns[@] instanceof SQLInteger) { 
SQLInteger sInt = (SQLInteger) anns[6] ; 
J] Use field name if name not specified 
if(sInt.name() .tength() < 1) 
columnName = field.getName().toUpperCase(); 
else 
columnName = sInt.name(); 
columnDefs.add(columnName + " INT" + 
getConstraints(sInt.constraints())); 


if(anns[6] instanceof SQLString) { 

SQLString sString = (SQLString) anns[@]; 
// Use field name if name not specified. 
if (sString.name().length() < 1) 

columnName = field. getName() .toUpperCase(); 
else 

columnName = sString.name(); 
columnDefs.add(columnName + " VARCHAR(" + 
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sString.value() + ")" + 
getConstraints (sString.constraints()));, 


StringBuilder createCommand = new StringBuilder ( 
"CREATE TABLE ”+ tableName + "("); 

for(Strifg columnDef : columnDefs) 
createCommand.append("\n " + columnDef + ","): 

77 Remove trailing comma 

String tableCreate = createCommand. substring( 
©, createCommand.tength() - 1) + "):": 

System.out.printin("Table Creation SQL for “ + 
className + " is :\n" + tableCreate); 

$ 
} 


i 2 
private static String getConstraints(Constraints con) { 
String constraints = "*; 
if(!con.allowNull ()) 
constraints += " NOT NULL"; 
if (con. primaryKey()) 
constraints += " PRIMARY KEY": 
if (con. unique()) 
constraints += " UNIQUE"; 
return constraints; 


} 

} /* Output: 
Table Creation SQL for annotations.database.Member 
CREATE TABLE MEMBER( 

FIRSTNAME VARCHAR (30)) ; 
Table Creation SQL for annotations.database.Member is : 
CREATE TABLE MEMBER( 

FIRSTNAME VARCHAR (30), 

LASTNAME VARCHAR(5O)) ; 
Table Creation SQL for annotations.database.Member is : 
CREATE TABLE MEMBER( 

FIRSTNAME VARCHAR (30). 

LASTNAME VARCHAR(59) , $ 

AGE INT); 
Table Creation SQL for annotations.database.Member is : 
CREATE TABLE MEMBER( 

FIRSTNAME VARCHAR(30) , 

LASTNAME VARCHAR (50). 

AGE INT, 

HANDLE VARCHAR(30) PRIMARY KEY) ; 
hm 


main() 方 法 会 处 理 命令 行 传人 的 每 一 个 类 名 。 使 用 forName() 方 法 加 载 每 一 个 类 ， 并 使 用 
getAnnotation(DBTable.class) 检 查 该 类 是 否 带 有 @DBTable 注 解 。 如 果 有 ， 就 将 发 现 的 表 名 保存 下 
来 。 然 后 读 取 这 个 类 的 所 有 域 ， 并 用 getDeclaredAnnotation0 进 行 检查 。 该 方法 返回 一 个 包含 一 个 
域 上 的 所 有 注解 的 数组 。 最 后 用 instanceof 操 作 符 来 判断 这 些 注解 是 否 是 @SQLIntege 或 
@SQLString 类 型 ， 如 果 是 的 话 ， 在 对 应 的 处 理 块 中 将 构造 出 相应 cloumn 名 的 字符 串 片 断 。 注 意 ， 
由 于 注解 没有 继承 机 制 ， 所 以 要 获得 近似 多 态 的 行为 ， 使 用 getDeclaredAnnotation0 是 唯一 的 办 法 。 

供 套 中 的 @Constraint 注 解 被 传递 给 getConstraints0 方 法 ， 由 它 负责 构造 一 个 包含 SQL 约束 
的 String 对 象 。 

需要 提醒 读者 的 是 ， 上 面 演示 的 技巧 对 于 真实 的 对 象 /关系 映射 而 言 ， 是 很 幼稚 的 。 例 如 使 
用 @DBTable 类 型 的 注解 ， 程 序 员 以 参数 的 形式 给 出 表 的 名 字 ， 如 果 程 序 员 想 要 修改 表 的 名 字 ， 
这 将 迫使 其 必须 重新 编译 Java 代 码 。 这 可 不 是 我 们 希望 看 到 的 结果 。 现 在 已 经 有 了 很 多 可 用 的 
framework， 可 以 将 对 象 映射 到 关系 数据 库 ， 并 且 ， 其 中 越 来 越 多 的 framework 已 经 开始 利用 注 
解 了 。 
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练习 1: 2) 为 本 节 数 据 库 的 例子 实现 更 多 的 SQL 类 型 。 
作业 。 : 修改 数据 库 的 例子 ， 使 其 能 够 使 用 JDBC 连 接 到 一 个 真正 的 数据 库 ， 并 与 之 交互 。 
作业 : 修改 数据 库 的 例子 ， 令 其 生成 XML 构 造 文件 ， 而 不 是 SQL 语 句 。 


20.3 使 用 apt 处 理 注解 


注解 处 理工 具 apt， 这 是 Sun 为 了 帮助 注解 的 处 理 过 程 而 提供 的 工具 。 由 于 这 是 该 工具 的 第 
一 版 ， 其 功能 还 比较 基础 ， 不 过 它 确实 有 助 于 程序 员 的 开发 工作 。 

与 javac 一 样 ，apt 被 设计 为 操作 Java 源 文件 ， 而 不 是 编译 后 的 类 。 默 认 情 况 下 ，apt 会 在 处 
理 完 源 文件 后 编译 它们 。 如 果 在 系统 构建 的 过 程 中 会 自动 创建 一 些 新 的 源 文件 ， 那 么 这 个 特性 
非常 有 用 。 事 实 上 ，apt 会 检查 新 生成 的 源 文件 中 注解 ， 然 后 将 所 有 文件 一 同 编译 。 

当 注 解 处 理 器 生成 一 个 新 的 源 文 件 时 ， 该 文件 会 在 新 一 轮 (round，Sun 文 档 中 这 样 称呼 它 ) 
的 注解 处 理 中 接受 检查 。 该 工具 会 一 轮 一 轮 地 处 理 ， 直 到 不 再 有 新 的 源 文 件 产生 为 止 。 然 后 它 
再 编译 所 有 的 源 文件 。 

程序 员 自 定 义 的 每 一 个 注解 都 需要 自己 的 处 理 器 ， 而 apt 工 具 能 够 很 容易 地 将 多 个 注解 处 理 

， 器 组 合 在 一 起 。 有 了 它 ， 程 序 员 就 可 以 指定 多 个 要 处 理 的 类 ， 这 上 比 程序 员 自 己 遍历 所 有 的 类 文 
件 简单 多 了 。 此 外 还 可 以 添加 监听 器 ， 并 在 一 轮 注解 处 理 过 程 结束 的 时 候 收 到 通知 信息 。 

在 撰写 本 章 的 时 候 ，apt 还 不 是 一 个 正式 的 Ant 任 务 (参见 http://MindView.net/Books/ 
BetterJava 中 的 附件 )， 不 过 显然 可 以 将 其 作为 一 个 Ant 的 外 部 任务 运行 。 要 想 编译 这 一 节 中 出 现 
的 注解 处 理 器 ， 你 必须 将 tools.jar 设 置 在 你 的 classpath 中 ， 这 个 工具 类 库 同 时 还 包含 了 
com.sun.mirror.* 接 口 。 

通过 使 用 AnnotationProcessorFactory，apt 能 够 为 每 一 个 它 发 现 的 注解 生成 一 个 正确 的 注解 
处 理 器 。 当 你 使 用 apt 的 时 候 ， 必 须 指明 一 个 工厂 类 ， 或 者 指明 能 找到 apt 所 需 的 工厂 类 的 路 径 。 
否则 ，apt 会 踏 上 一 个 神秘 的 探索 之 旅 ， 详 细 的 信息 可 以 在 Sun 文 档 的 “开发 一 个 注解 处 理 器 ” 
一 节 中 找到 。 

使 用 apt 生 成 注解 处 理 器 时 ， 我 们 无 法 利用 Java 的 反射 机 制 ， 因 为 我 们 操作 的 是 源 代码 ， 而 
不 是 编译 后 的 类 8 。 使 用 mirror API® 能 够 解决 这 个 问题 ， 它 使 我 们 能 够 在 未 经 编译 的 源 代码 中 
查看 方法 ， 域 以 及 类 型 。 

下 面 是 一 个 自 定义 的 注解 ， 使 用 它 可 以 把 一 个 类 中 的 public 方 法 提取 出 来 ， 构 造成 一 个 新 的 
接口 ; 


//: annotations/ExtractInterface. java 

// APT-based annotation processing. ` 

package annotations; 4 

import java.lang.annotation.*; ` 

@Target (ElementType. TYPE) 

@Retention(RetentionPol icy. SOURCE) 

public @interface ExtractInterface { 
public String value(); 

} :~ 


RetentionPolicy 是 SOURCE， 因 为 当 我 们 从 一 个 使 用 了 该 注解 的 类 中 抽取 出 接口 之 后 ， 没 
有 必要 再 保留 这 些 注解 信息 。 下 面 的 类 有 一 个 公共 方法 ， 我 们 将 会 把 它 抽 取 到 一 个 有 用 接口 中 ， 


O 作业 ， 我 建议 读者 将 其 作为 课程 大 作业 。 解 答 指南 中 不 包含 此 类 作业 的 解决 方案 。 
© 不 过 ， 使 用 非 标准 的 选项 -XelassesAsDecls， 你 可 以 在 编译 后 的 类 中 操作 注解 。 
© Java 设 计 师 们 卖弄 地 认为 镜子 (mirror) 就 是 起 反射 (reflection) 的 作用 。 
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/1: annotations/Multiplier.java 
// APT-based annotation processing. 
package annotations; 


@ExtractInterface("IMultiplier”) 
public class Multiplier { 
public int multiply(int x, int y) { 
int total = @; 
for(int i = 0; i < x; i++) 
total = add(total, y); 
return total; 
} 
private int add(int x, int y) { return x + y; } 
public static void main(String[] args) { 
Multiplier m = new Multiplier(): ` 
System.out.printin("11*16 = ”+ m.multiply(21, 16)): 
} 





} /* Output: 
11°16 = 176 
7A 





在 Multiplier 类 中 〈 它 只 对 正 整 数 起 作用 ) 有 一 个 multiply0 方 法 ， 该 方法 多 次 调用 一 个 私有 


的 add0 方 法 以 实现 乘法 操作 。add0 方 法 不 是 公共 的 ， 因 此 不 将 其 作为 接口 的 一 部 分 。 注 解 给 出 
了 值 IMultiplier， 这 就 是 将 要 生成 的 接口 的 名 字 : 


//: annotations/InterfaceExtractorProcessor. java 
// APT-based annotation processing. 

// {Exec: apt -factory 

// annotations. Inter faceExtractorProcessorFactory 
// Multiplier.java -s ../annotations} 

package annotations; 

import com.sun.mirror.apt.*; 

import com.sun.mirror.declaration.*; 

import java.io.*; 

import java.util 





public class InterfaceExtractorProcéssor 
implements AnnotationProcessor { 
private final AnnotationProcessorEnvironment env: 
private ArrayList<MethodDeclaration> interfaceMethods = 
new ArrayList<MethodDeclaration>(); 
public Interface€xtractorProcessor( 
AnnotationProcessorEnvironment env) { this.env = env; } 
public void process() { 
for (TypeDeclaration typeDecl : 
env. getSpecifiedTypeDeclarations()) { 
ExtractInterface annot = 
typeDect.getAnnotation(ExtractInterface.class) ; 
if(annot == null) 
break: 
for(MethodDeclaration m :*typeDecl.getMethods()) 
if (m.getModifiers() contains (Modifier .PUBLIC) && 
!(m. getModi fiers() .contains (Modifier. STATIC))) 
interfaceMethods . add(m) ; 
if(interfaceMethods.size() > 0) { 
try { 
PrintWriter writer = 
env. getFiler() .createSourceFile(annot.. value()): 
writer.printin("package ”+ 
typeDecl. getPackage() .getQualifiedName() +";") 
writer.printin("public interface ”+ 
annot.value() + * {"); 
for (MethodDeclaration m interfaceMethods) { 
writer.print(" public "); 
writer.print(m.getReturnType() + " * 
writer.print(m.getSimpleName() +" ("); 
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int 1 = 0; 
for(ParameterDeclaration parm : 
m.getParameters()) { 
writer.print(parm.getType() +" " + 
parm. getSimpleName()); 
if (++i < m.getParameters().size()) 
writer.print(*, "); 


weiter.printin("):"); 

t 
writer.printin("}"): 
writer.close(); 

} catch(IOException ioe) { 
throw new Runt imeException(ioe); 

i 

} 
} 


} 

} Mi~ 

所 有 的 工作 都 在 process0 方 法 中 完成 。 在 分 析 一 个 类 的 时 候 ， 我 们 用 MethodDeclaration 类 以 
及 其 上 的 getModifiers0 方 法 来 找到 public 方 法 (不 包括 static 的 那些 )。 一 旦 找到 我 们 所 需 的 publie 
方法 ， 就 将 其 保存 在 一 个 ArrayList 中 ， 然 后 在 一 个 java 文 件 中 ， 创 建新 的 接口 中 的 方法 定义 。 

注意 ， 处 理 器 类 的 构造 器 以 AnnotationProcessorEnvironment 对 象 为 参数 。 通 过 该 对 象 ， 我 
们 就 能 知道 apt 正 在 处 理 的 所 有 类 型 (类 定义 )， 并 且 可 以 通过 它 获 得 Messager 对 象 和 Filer 对 象 。 
Messager 对 象 可 以 用 来 向 用 户 报告 信息 ， 比 如 处 理 过 程 中 发 生 的 任何 错误 ， 以 及 错误 在 源 代 码 
中 出 现 的 位 置 等 。Filer 是 一 种 PrintWriter， 我 们 可 以 通过 它 创建 新 的 文件 。 不 使 用 普通 的 
了 PrintWriter 而 使 用 Filer 对 象 的 主要 原因 是 ， 只 有 这 样 apt 才 能 知道 我 们 创建 的 新 文件 ， 从 而 对 新 
文件 进行 注解 处 理 ， 并 且 在 需要 的 时 候 编 译 它们 。 : 

同时 我 们 看 到 ，Filer 的 createSourceFile( 方 法 以 将 要 新 建 的 类 或 接口 的 名 字 ， 打 开 了 一 个 
普通 的 输出 流 。 现 在 还 没有 什么 工具 帮助 程序 员 创建 Java 语 言 结构 ， 所 以 我 们 只 能 用 基本 的 
printO0 和 printin( 方 法 来 生成 Java 源 代码 。 因 此 ， 你 必须 小 心 仔 细 地 处 理 括号 ， 确 保 其 闭合 ， 并 
且 确 保生 成 的 代码 语法 正确 。 

apt 工 具 需 要 一 个 工厂 类 来 为 其 指明 正确 的 处 理 器 ， 然 后 它 才能 调用 处 理 器 上 的 process() 
Bik: 


//: annotat ions/InterfaceExtractorProcessorFactory. java 
// APT-based annotation processing. 

package annotations; 

import com.sun.mirror.apt.*; 

import com.sun.mirror.declaration.*; 

import java.util.*; 


public class InterfaceExtractorProcessorFactory 
implements AnnotationProcessorFactory { 
public AnnotationProcessor getProcessorFor( 
Set<AnnotationTypeDeclaration> atds, 
AnnotationProcessorEnvironment env) { 
return new InterfaceExtractorProcessor (env): 


} 

public Collection<String> supportedAnnotationTypes() { 
return 
Collections. singleton("annotations.€xtractInterface") ; 


} 

public Collection<String> supportedOptions() { 
return Collections.emptySet(); 

} 

H~ 
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AnnotationProcessorFactory 接 口 只 有 三 个 方法 。 如 你 所 见 ， 其 中 之 一 的 getProcessorFor0 方 
法 返回 注解 处 理 器 ， 该 方法 以 包含 类 型 声明 的 Set (使 用 apt 工 具 时 传人 的 Java 类 ) 以 及 
AnnotationProcessorEnvironment 对 象 为 参数 (将 传人 给 处 理 器 对 象 ) 。 另 外 两 个 方法 是 
SupportedAnnotationTypes0 和 supportedOptions0， 程 序 员 可 以 通过 它们 检查 一 下 ， 是 否 apt 工 具 
发 现 的 所 有 的 注解 都 有 相应 的 处 理 器 , 是 否 所 有 控制 台 输 入 的 参数 都 是 你 提供 支持 的 选项 。 其 中 ， 
supportedAnnotationTypes 0 方法 尤其 重要 ， 因 为 一 旦 在 返回 的 String 集 合 中 没有 你 的 注解 的 完整 
类 名 ，apt 就 会 抱怨 没有 找到 相应 的 处 理 器 ， 从 而 发 出 警告 信息 ， 然 后 什么 也 不 做 就 退出 。 

以 上 例子 中 的 处 理 器 与 工厂 类 都 在 annotations 包 中 ， 在 InterfaceExtractorProcessorjava 开 
头 的 注释 文字 中 ,我 根据 anotations 的 目录 结构 ,在 Exec 标 记 处 给 出 了 需要 从 命令 行 输入 的 命令 。 
它 将 告诉 apt 工 具 ， 使 用 上 面 的 工厂 类 来 处 理 Multiplierjava 文 件 。 参 数 -s 说 明 任何 新 产生 的 文件 
都 必须 放 在 annotations 目 录 中 。 通 过 处 理 器 中 的 printin0 语 句 ， 估 计 你 已 经 能 猜 到 最 终生 成 的 
IMultiplierjava 会 是 什么 样子 了 : 


package: annotations; 
public. interface IMultiplier { 
public int multiply (int x, int y); 
} 


apt 也 会 编译 这 个 新 产生 的 文件 ， 因 此 你 将 在 相同 的 目录 中 Fipe dakik 
练习 2: (3) 为 抽取 出 来 的 接 口 添 加 对 除法 的 支持 。 


20.4 将 观察 者 模式 用 于 apt 


上 面 的 例子 是 一 个 相当 简单 的 注解 处 理 器 ， 只 需 对 一 个 注解 进行 分 析 ， 但 我 们 仍然 要 做 大 
量 复 杂 的 工作 。 因 此 ， 处 理 注解 的 真实 过 程 可 能 会 非常 复杂 。 当 我 们 有 更 多 的 注解 和 更 多 的 处 
理 器 时 ， 为 了 防止 这 种 复杂 性 迅速 攀升 ，mirror API 提 供 了 对 访问 者 设计 模式 的 支持 。 访 问 者 
是 Gamma 等 人 所 著 的 《设计 模式 》8。 一 书 中 的 经 典 设计 模式 之 一 。 你 也 可 以 在 《Thinking in 
Pattems》 中 找到 更 详细 的 解释 。 

一 个 访问 者 会 遍历 某 个 数据 结构 或 一 个 对 象 的 集合 ， 对 其 中 的 每 一 个 对 象 执行 一 个 操作 。 
该 数据 结构 无 需 有 序 ， 而 你 对 每 个 对 象 执行 的 操作 ， 都 是 特定 于 此 对 象 的 类 型 。 这 就 将 操作 与 
对 象 解 焕 ， 也 就 是 说 ， 你 可 以 添加 新 的 操作 ， 而 无 需 向 类 的 定义 中 添加 方法 。 

这 个 技巧 在 处 理 注解 时 非常 有 用 ， 因 为 一 个 Java 类 可 以 看 作 是 一 系列 对 象 的 集合 ， 例 如 
TypeDeclaration 对 象 、FieldDeclaration 对 象 以 及 MethodDeclaration 对 象 等 。 当 你 配合 访问 者 模 
式 使 用 apt 工 具 时 ， 需 要 提供 一 个 Visitor 类 ， 它 具有 一 个 能 够 处 理 你 要 访问 的 各 种 声明 的 方法 。 
然后 ， 你 就 可 以 为 方法 、 类 以 及 域 上 的 注解 实现 相应 的 处 理 行为 。 

下 面 仍 然 是 SQL 表 生成 器 的 例子 ， 不 过 这 次 我 们 使 用 访问 者 模式 来 创建 工厂 和 注解 处 理 器 : 


//: annotations/database/TableCreationprocessorFactory.jaya 

// The database example using Visitor. 

/1 {Exec: apt -factory 

// annotations .database.TableCreat ionProcessorFactory 

// database/Member.java -s database} 

package annotations database: i 
import com.sun.mirror.apt.*: 4 
import com.sun.mirror.declaration.*; 

import com.sun.mirror.util.*; 

import java.util.*; Ş 

import static com.sun.mirror.util.DeclarationVisitors.*; 


O 本 书 中 文 版 、 英 文 版 以 及 双语 版 均 已 由 机 械 工业 出 版 社 出 版 一 编辑 注 。 
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public class TableCreationProcessorFactory 
implements AnnotationProcessorFactory { 
public AnnotationProcessor getProcessorFor( 
Set<AnnotationTypeDeclaration> atds, 
AnnotationProcessorEnvironment env) { 
return new TableCreationProcessor (env); 
} 
public Collection<String> supportedAnnotationTypes() { 
return Arrays.asList( 
“annotations. database.DBTable" , 
“annotations .database.Constraints”, 
“annotations.database. SQLString”, 
“annotations.database.SQLInteger"); 
} 
public Collection<String> supportedOptions() { 
return Collections.emptySet(); 
} 
private static class TableCreationProcessor 
implements AnnotationProcessor { 
private final AnnotattonProcessorEnvironment env; 
private String sql = ""; 
public TableCreationProcessor ( 
AnnotationProcessorEnvironment env) { 
this.env = env; 


public void process() { 
for(TypeDeclaration typeDecl : 

env.getSpecifiedTypeDeclarations()) { 
typeDecl. accept (getDeclarat ionScanner ( 
new TableCreationVisitor(), NO_OP)); 

sql = sql.substring(®, sqt.length() - 1) + “y; 

System.out.printin("creation SQL is :\n" + sql); 

sql =" 








} 
$ 
private class TableCreationVisitor 
extends SimpleDeclarationVisitor { 
public void visitClassDeclaration( 
ClassDeclaration d) { 
DBTable dbTable = d.getAnnotation(DBTable.class) ; 
if(dbTable != null) { 
sql += "CREATE TABLE "; 
sql += (dbTable.name().length() < 1) 
? d.getSimpleName(). toUpperCase() 
: dbTable.name() ; 
sql +=" ("; 
} 


} 
public void visitFieldDeclaration( 
FieldDeclaration d) { 
String columnName = ""; 
if (d.getAnnotation(SQLInteger.class) != null) { 
SQLInteger sInt = d.getAnnotation( 
SQLInteger.class) ; 
// Use field name if name not specified 
if (sInt.name().length() < 1) 
columnName = d.getSimpleName(). toUpperCase():; 





else 
columnName = sInt.name(); 
sql += "\n ”+ columnName + * INT" + 


getConstraints(sInt.constraints()) + "."; 
} 
if(d.getAnnotation(SQLString.class) != null) { 
SQLString sString = d.getAnnotation( 
SQLString.class) 
// Use field name if name not specified. 
if(sString.name().length() < 1) 
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columnName = d.getSimpleName().toUpperCase() ; 
else 
columnName = sString.name(); 
sql += "\n ”+ columnName + " VARCHAR(" + 
sString.value() + ")" + 
getConstraints(sString.constraints()) + ","; 


$ 


} 
private String getConstraints(Constraints con) { 
String constraints = ""; 
if(!con.allowNul1()) 
constraints += " NOT NULL"; 
if (con.primaryKey()) 
constraints += " PRIMARY KEY"; 
if (con.unique()) 
constraints += " UNIQUE"; 
return constraints; 
$ 
} 


} 

) Mi~ 

这 个 程序 输出 的 结果 与 前 一 个 DBTable 的 例子 完全 相同 。 

在 这 个 例子 中 ， 处 理 器 与 访问 者 都 是 内 部 类 。 注 意 ，process0 方 法 所 做 的 只 是 添加 了 一 个 
访问 者 类 ， 并 初始 化 了 SQL 字符 串 。 

getDeciarationScanner( 方 法 的 两 个 参数 都 是 访问 者 ;第 一 个 是 在 访问 每 个 声明 前 使 用 ， 第 二 
个 则 是 在 访问 之 后 使 用 。 由 于 这 个 处 理 器 只 需要 在 访问 前 使 用 的 访问 者 ， 所 以 第 二 个 参数 给 的 是 
NO_OP。NO_OP 是 DeclarationVisitor 接 口中 的 static 域 ， 是 一 个 什么 也 不 做 的 Declaration-Visitor。 

TableCreationVisitor 继 承 自 SimpleDeclarationVisitor， 它 覆 写 了 两 个 方法 visitClase- 
Declaration() 和 visitFieldDeclaration()。SimpleDeclarationVisitor 是 一 个 适配器 ， 实 现 了 
DeclarationVisitor 接 口中 的 所 有 方法 ， 因 此 ， 程 序 员 只 需 将 注意 力 放 在 自己 需要 的 那些 方法 上 。 
在 visitClaseDeclaration() 方 法 中 ， 检 查 ClassDeclaration 对 象 是 否 带 有 DBTable 注 解 ， 如 果 存 在 
的 话 ， 将 初始 化 SQL 语句 的 第 一 部 分 。 在 visitFieldDeclaration0 方 法 中 ， 将 检查 域 声明 上 的 注解 ， 
从 域 声明 中 提取 信息 的 过 程 与 本 章 前 面 的 例子 一 样 。 

看 起 来 这 个 例子 使 用 的 方式 似乎 更 复杂 ， 但 是 它 确实 是 一 种 具备 扩展 能 力 的 解决 方案 。 当 
你 的 注解 处 理 器 的 复杂 性 越 来 越 高 的 时 候 ， 如 果 还 按 前 面 例子 中 的 方式 编写 自己 独立 的 处 理 器 ， 
那么 很 快 你 的 处 理 器 就 将 变 得 非常 复杂 。 

练习 3: (2) 向 TableCreationProcessorFactory.java 中 添加 对 更 多 的 SQL 类 型 的 支持 。 


20.5 基于 注解 的 单元 测试 


单元 测试 是 对 类 中 的 每 个 方法 提供 一 个 或 多 个 测试 的 一 种 实践 ， 其 目的 是 为 了 有 规律 地 测 
试 一 个 类 的 各 个 部 分 是 否 具备 正确 的 行为 。 在 Java 中 ， 最 著名 的 单元 测试 工具 就 是 JUnit。 在 撰 
写本 书 时 ，JUnit 已 经 开始 了 向 JUnit4 更 新 的 过 程 ， 其 目的 正 是 为 了 融入 注解 x。 对 于 注解 出 现 之 
前 的 JUnit 而 言 ， 有 一 个 主要 的 问题 ， 即 为 了 设置 并 运行 JUnit 测 试 需要 做 大 量 的 形式 上 的 工作 。 
随 着 其 渐渐 的 发 展 ， 这 种 负担 已 经 减轻 了 一 些 ， 但 注解 的 出 现 能 够 使 其 更 贴近 “最 简单 的 单元 
测试 系统 ”。 

使 用 注解 出 现 之 前 的 JUnit， 程 序 员 必 须 创建 一 个 独立 的 类 来 保存 其 单元 测试 。 有 了 注解 ， 


O 我 原本 考虑 过 基于 这 里 的 设计 来 自己 做 一 个 Better JUnit, 不 过 后 来 发 现 JUnit4 已 经 具有 很 多 我 这 里 讲 到 的 思想 ， 
因此 直接 升级 为 JUnit4 可 能 更 简单 吧 。 
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我 们 可 以 直接 在 要 验证 的 类 里 面 编写 测试 ， 这 将 大 大 减少 单元 测试 所 需 的 时 间 和 麻烦 之 处 。 采 

用 这 种 方式 还 有 一 个 额外 的 好 处 ， 就 是 能 够 像 测试 public 方 法 一 样 很 容易 地 测试 private 方 法 。 1083) 
这 个 基于 注解 的 测试 框架 叫做 @Unit。 其 最 基本 的 测试 形式 ， 可 能 也 是 你 用 的 最 多 的 一 个 

注解 是 @Test， 我 们 用 @Test 来 标记 测试 方法 。 测 试 方法 不 带 参数 ， 并 返回 boolean 结 果 来 说 明 测 

试 成 功 或 失败 。 程 序 员 可 以 任意 命名 他 的 测试 方法 。 同 时 ，@Unit 测 试 方法 可 以 是 任意 你 喜欢 

的 访问 修饰 方式 ， 包 括 private。 
要 使 用 @Unit， 程 序 员 必须 引入 netmindview.atunits ， 用 @Unit 的 测试 标记 为 合适 的 方法 

和 域 打上 标记 (在 接 下 来 的 例子 中 你 会 学 到 ) ， 然 后 让 你 的 构建 系统 对 编译 后 的 类 运行 @Unit。 

下 面 是 一 个 简单 的 例子 ， 














//: annotations/AtUnitExamplel. java 
package annotations; 

import net.mindview.atunit.*: 
import net.mindview.util.*; 


public class AtUnitExamplel { 
public String methodOne() { 
return "This is methodOne"; 


$ 

public int methodTwo() { 
System.out.println("This is methodTwo") ; 
return 2; 


} 
@Test boolean methodOneTest() { 
return methodOne().equals("This is methodOne"); 


} 
@Test boolean m2() { return methodTwo() == 2; } 
@Test private boolean m3() { return true; } 
// Shows output for failure: 
@Test boolean failureTest() { return false; } 
@Test boolean anotherDisappointment() { return false: } 
public static void main(String{] args) throws Exception { 
OSExecute. command ( 
“java net.mindview.atunit.AtUnit AtUnitExamplel"); 


t 
} /* Output: 
annotations. AtUni tExamplel 
~ methodOneTest 
+ m2 This is methodTwo 


5| 
g 


m3 

. failureTest (failed) 

. anotherDisappointment (failed) 
(5 tests) 


>>> 2 FAILURES <<< 
annotations .AtUnitExamplel: failureTest 
annotations .AtUnitExamplel: anotherDisappointment 
Whim 


使 用 @Unit 进 行 测试 的 类 必须 定义 在 某 个 包 中 ( 即 必须 包括 packae 声 明 )。 

@Test 注 解 被 置 于 methodOneTest0、m20、m30failureTest0 以 及 anotherDisappointment() 
方法 之 前 ， 它 告诉 @Unit 将 这 些 方法 作为 单元 测试 来 运行 。 同 时 ，@Test 将 验证 并 确保 这 些 方 法 
没有 参数 ， 并 且 返 回 值 是 boolean 或 void。 程 序 员 编写 单元 测试 时 ， 唯一 需要 做 的 就 是 决定 测试 
是 成 功 还 是 失败 ，( 对 于 返回 值 为 boolean 的 方法 ) 应 该 返回 ture 还 是 false。 

如 果 你 熟悉 JUnit， 你 会 注意 到 @Unit 的 输出 带 有 更 多 的 信息 。 我 们 可 以 看 到 当前 正在 运行 


O 这 个 类 库 是 本 书 附带 的 代码 包 的 一 部 分 ， 可 以 在 www MindView net 上 找到 。 
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的 测试 ， 因 此 测试 中 的 输出 更 是 有 用 ， 而 且 在 最 后 ， 它 还 能 告诉 我 们 导致 错误 的 类 和 测试 。 
程序 员 并 非 必 须 将 测试 方法 伐 入 到 原本 的 类 中 ， 因 为 有 时 候 这 根本 做 不 到 。 要 生成 一 个 非 
嵌入 式 的 测试 ， 最 简单 的 办 法 就 是 继承 : 


//: annotations/AtUnitExternalTest. java 
// Creating non-embedded tests. 

package annotations; 

import net.mindview.atunit.*; 

import net.mindview.util.*: 


public class AtUnitExternalTest extends AtUnitExamplel {, 
@Test boolean _methodOne() { 
return methodOne().equals("This is methodOne"); 
} 
@Test boolean methodTwo() { return methodTwo() == 2: } 
public static void main(String[} args) throws Exception { 
1085] OSExecute. command ( 
"java net.mindview.atunit.AtUnit AtUnitExternalTest"); 
} 
} /* Output: 
annotations .AtUnitExternalTest 
. _methodone 
. LmethodTwo This is methodTwo 














OK (2 tests) 
Whim 


这 个 例子 还 表现 出 了 灵活 命名 的 价值 (与 JUnit 不 同 ， 它 要 求 你 必须 使 用 test 作 为 测试 方法 的 
前 缀 )。 在 这 里 ，@Test 方 法 被 命名 为 下 划 线 前 级 加 上 这 将 要 测试 的 方法 的 名 字 (我 并 不 认为 这 
是 一 个 理想 的 命名 形式 ， 只 是 表现 一 种 可 能 性 罢了 ) 。 

或 者 你 还 可 以 使 用 组 合 的 方式 创建 非 嵌入 式 的 测试 ; 


/1: annotations/AtUnitComposition. java 
// Creating non-embedded tests. 
package annotations; 

import net.mindview.atunit.*; 

import net.mindview.util.*; 


public class AtUnitComposition { 
AtUnitExamplel testObject = new AtUnitExamplel(); 
@Test boolean _methodOne() { 
return 
testObject .method0ne().equals("This is methodOne”) ; 


} 

@Test boolean _methodTwo() { 
return testObject.methodTwo() == 2; 

} 

public static void main(String[] args) throws Exception { 
OSExecute. command ( 
"java net.mindview.atunit.AtUnit AtUnitComposition"): 


} 
} /* Output: 
annotations. AtUnitComposit ion 
. _methodOne 
+ LmethodTwo This is methodTwo 


OK (2 tests) ‘ 
mog hx 
因为 每 个 测试 对 应 一 个 新 创建 的 AtUnitComposition 对 象 ， 因 此 每 个 测试 也 对 应 一 个 新 的 成 
员 testObject。 
@Unit 中 并 没有 JUnit 里 的 特殊 的 assert 方 法 ， 不 过 @Test 方 法 仍然 允许 程序 员 返 回 void (如 
果 你 还 是 想 用 ture 或 false 的 话 ， 你 仍然 可 以 用 boolean 作 为 方法 返回 值 类 型 ) ， 这 是 @Test 方 法 的 
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第 二 种 形式 。 在 这 种 情况 下 ， 要 表示 测试 成 功 ， 可 以 使 用 Java 的 assert 语 句 。Java 的 断言 机 制 一 
般 要 求 程序 员 在 java 命 令 行 中 加 上 -ea 标志 ， 不 过 @Unit 已 经 自动 打开 了 该 功能 。 而 要 表示 测试 
失败 的 话 ， 你 甚至 可 以 使 用 异常 。@Unit 的 设计 目标 之 一 就 是 尽 可 能 少 地 添加 额外 的 语法 ， 而 
Java 的 assert 和 异常 对 于 报告 错误 而 言 ， 已 经 足够 了 。 一 个 失败 的 assert 或 从 测试 方法 中 抛 出 异 
常 ， 都 将 被 看 作 一 个 失败 的 测试 ， 但 是 @Unit 并 不 会 就 在 这 个 失败 的 测试 上 打住 ， 它 会 继续 运 
行 ， 直到 所 有 的 测试 都 运行 完毕 。 下 面 是 一 个 示例 程序 : 


//: annotations/AtUnitExample2. java 
// Assertions and exceptions can be used in @Tests. 
package annotations; 

import java.io.*; 

import net.mindview.atunit.*; 

import net.mindview.util.*; 


public class AtUnitExample2 { 
public String methodOne() { 
return “This is methodOne"; 


} 

public int methodTwo() { 
System.out.printin("This is methodTwo"); 
return 2; 


} 
@Test void assertExample() { 
assert methodOne().equals("This is methodOne"); 


} 

@Test void assertFailureexample() { 
assert 1 == 2: "What a surprise!"; 

} 

@Test void exceptionExample() throws IOException { 
new FileInputStream("nofile.txt"); // Throws 


@Test boolean assertAndReturn() { 
// Assertion with message: 
assert methodTwo() == 2: "methodTwo must equal 2"; 
return methodOne().equals("This is methodOne”) ; 








} 
public static void main(String(} args) throws Exception { 
OSExecute. command ( 
“java net.mindview.atunit.AtUnit AtUnitExample2"); 


} 
了 /* Output: 
annotations .AtUnitExample2 

. assertExample 

. assertFailureExample java. lang.Assertion€rror: What a 
surprise! 

(failed) 

. exceptionExample java. io.FileNotFoundException: 
nofile.txt (The system cannot find the file specified) 
(failed) 

. assertAndReturn This is methodTwo 


(4 tests) 


>>> 2 FAILURES <<< 
annotattons.AtUnit€xample2: assertFailureExample 
annotations .AtUnitExample2: except ionExample 
Wh 


下 面 的 例子 使 用 非 代 人 式 的 测试 ， 并 且 用 到 了 断言 ， 它 将 对 javautilLHashSet 执 行 一 些 简单 
的 测试 : 


//: annotations/HashSetTest.java 
package annotations; 








[oa 





638 #20% 











11089] 











import java.util. *; 
import net.mindview.atunit.*; 
import net.mindview.util.*; 


public class HashSetTest { 
HashSet<String> testObject = new HashSet<String>(); 
@Test void initialization() { 
assert testObject.isEmpty(); 


} 

@Test void _contains() { 
testObject.add ("one"); 
assert testObject.contains("one"); 

} 

@Test void _remove() { 
testObject.add("one") ; 
testObject.remove("one"): 
assert testObject. isEmpty(); 

} 

public static void main(String[] args) throws Exception { 
OSExecute. command ( 

"java net.mindview.atunit.AtUnit HashSetTest"); 

} 

} /* Output: 
annotations .HashSetTest 

. initialization 

. remove 

+ contains 

OK (3 tests) 
Whim 


如 果 采 用 继承 的 方式 ， 可 能 会 更 简单 ， 并 且 也 没有 一 些 其 他 的 约束 。 


练习 4: (3) 验证 是 否 每 个 测试 都 会 生成 一 个 新 的 testObject。 
练习 5: (1) 使 用 继承 的 方式 修改 上 面 的 例子 。 

练习 6: (1) 使 用 HashSetTestjava 演 示 的 方式 测试 LinkedList 类 。 
练习 7: (1) 使 用 继承 的 方式 修改 前 一 个 练习 的 结果 。 

对 每 一 个 单元 测试 而 言 ，@Unit 都 会 用 默认 的 构造 器 ， 为 该 测试 所 属 的 类 创建 出 一 个 新 的 


实例 。 并 在 此 新 创建 的 对 象 上 运行 测试 ， 然 后 丢弃 该 对 象 ， 以 避免 对 其 他 测试 产生 副作用 。 如 
此 创建 对 象 导致 我 们 依赖 于 类 的 默认 构造 器 。 如 果 你 的 类 没有 默认 构造 器 ， 或 者 新 对 象 需要 复 
杂 的 构造 过 程 ， 那 么 你 可 以 创建 一 个 static 方 法 专门 负责 构造 对 象 ， 然 后 用 @TestObjectCreaet 注 
解 将 该 方法 标记 出 来 ， 就 像 这 样 : 


//: annotations /AtUnitExample3. java 
package annotations; 

import net.mindview.atunit.*; 
import net.mindview.util.*; 


public class AtUnitéxample3 ( 

private int ni 

public AtUnitExample3(int n) { this.n = n; } 

public int getN() { return n; } 

public String methodOne() { 
return "This is methodOne"; 

} 

public int methodTwo() { 
System.out.printin("This is methodTwo"); 
return 2; 


} 
@TestObjectCreate static AtUnitExample3 create() { 
return new AtUnitExample3(47); 


} 
@Test boolean initialization() { return n == 
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@Test boolean methodOneTest() { 
return methodOne().equals("This is methodOne"); 





} 
@Test boolean m2() { return methodTwo() == 2; } 
public static void main(String[] args) throws Exception { 
OSExecute. command ( 
"java net.mindview.atunit.AtUnit AtUnitexample3") ; 
} 
} /* Output: 
annotations.AtUnitExample3 
. initialization 
+ methodOneTest 
. m2 This is methodTwo 


Ok (3 tests) 
Whim 


加 入 了 @TestObjectCreaet 注 解 的 方法 必须 声明 为 static， 且 必须 返回 一 个 你 正在 测试 的 类 型 
的 对 象 ， 这 一 切 都 由 @Unit 负 责 确保 成 立 。 : 

有 的 时 候 ， 我 们 需要 向 单元 测试 中 添加 一 些 额外 的 域 。 这 时 可 以 使 用 @TestProperty 注 解 ， 
由 它 注解 的 域 表 示 只 在 单元 测试 中 使 用 〈 因 此 ， 在 我 们 将 产品 发 布 给 客户 之 前 ， 他 们 应 该 被 删 
除 掉 )。 在 下 面 的 例子 中 ， 一 个 String 通 过 String.split0 方 法 被 拆散 了 ， 从 其 中 读 取 一 个 值 ， 这 个 
值 将 被 用 来 生成 测试 对 象 : 


//: annotations/AtUnitExample4. java 
package annotation 
import java.util.*; 
import net.mindview.atunit.*; 
import net.mindview.util. 
import static net.mindview.util.Print.*; 1090) 
public class AtUnitExample4 { 
static String theory = "All brontosauruses "+ 
"are thin at one end, much MUCH thicker in the * + 
"middle, and then thin again at the far end.”; 
private String word; 
Private Random rand = new Random(); // Time-based seed 
public AtUnitExample4 (String word) { this.word = word: } 
public String getWord() { return word: } 
public String scrambleWord() { 
List<Character> chars = new ArrayList<Character>(); 
for(Character c : word.toCharArray()) 
chars.add(c); 
Collections. shuffle(chars, rand); 
StringBuilder result = new StringBuilder (); 
for(char ch : chars) 
result. append(ch) ; 
return result.toString(); 




















} 
@TestProperty static List<String> input = 
Arrays.asList(theory.split(" ")); 
@TestProperty 
static Iterator<String> words = input. iterator (); 
@TestObjectCreate static AtUnitexampled create() { 
if (words. hasNext ()) 
return new AtUnitExample4(words.next()); 
else 
return null; 
} 
@Test boolean words() { 
print("'" + getWord() + "'"); 
return getWord().equals("are 





} 
@Test boolean scramblel() { 
// Change to a specific seed to get verifiable results: 
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rand = new Random(47); 

print("'" + getWord() + "'"); 
String scrambled = scrambleWord() ; 
print(scrambled) ; 

return scrambled.equals(*1Al"); 


} 
@Test boolean scramble2() { 
rand = new Random(74) ; 
print("'" + getWord() + "'"): 
String scrambled = scrambleWord(); 
print (scrambled) ; 
return scrambled. equals("tsaeborornussu"): 


$ 
public static void main(String[] args) throws Exception { 
System.out.printin("starting"); 
OSExecute. command ( 
“java net.mindview.atunit.AtUnit AtUnitExample4") ; 


} 

} /* Output: 

starting 

annotations .AtUnitExample4 
scramblel ‘ALI’ 

1Al 


. scramble2 'brontosauruses' 
tsaeborornussu 


. words ‘are’ 


OK (3 tests) 
Wham 


@TestProperty 也 可 以 用 来 标记 那些 只 在 测试 中 使 用 的 方法 ， 而 他 们 本 身 又 不 是 测试 方法 。 

注意 ， 这 个 程序 依赖 于 测试 执行 的 顺序 ， 这 可 不 是 一 个 好 的 实践 。 

如 果 你 的 测试 对 象 需要 执行 某 些 初始 化 工作 ， 并 且 使 用 完毕 后 还 需要 进行 某 些 清理 工作 ， 
那么 可 以 选择 使 用 static @TestObjectCleanup 方 法 ， 当 测试 对 象 使 用 结束 后 ， 该 方法 会 为 你 执行 
清理 工作 。 在 下 面 的 例子 中 ，@TestObjectCreate 为 每 个 测试 对 象 打开 了 一 个 文件 ， 因 此 必须 在 
丢弃 测试 对 象 的 时 候 关 闭 该 文件 : 


//: annotations/AtUnitExampte5. java 
package annotations; 

import java.io.*; 

import net.mindview.atunit. 
import net.mindview.util.*; 








public class AtUnitExampte5 { 
private String text: 
public AtUnitExamples(String text) { this.text = text; } 
public String toString() { return text; } 
@TestProperty static PrintWriter output; 
@TestProperty static int counter; 
@TestObjectCreate static AtUnitExamples create() { 
String id = Integer. toString(counter++) ; 
try { 
output = new PrintWriter("Test" + id + ".txt"); 
} catch(IOException e) { 
throw new Runt imeException(e) ; 
} 
return new AtUnitExampleS(1d); 


} 

@TestObjectCleanup static void 

cleanup(AtUnitexamples tobj) { 
System.out .printtn("Running cleanup"); 
output .close(); 
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} 

@Test boolean test1() { 
output. print("test1"); 
return true; 

} 

@Test boolean test2() { 
output. print (*test2 
return true; 

} 

@Test boolean test3() { 
output.print("test3"): 
return true; 





public static void main(String[] args) throws Exception { 
OSExecute. command ( 
“java net.mindview.atunit.AtUnit AtUnitExamples") ; 

} 
} /* Output: 
annotations .AtUnitExampleS ‘ 

. testl 
Running cleanup 

. test2 
Running cleanup 

. test3 
Running cleanup 
OK (3 tests) 
"~ 


从 输出 中 我 们 可 以 看 到 ， 清 理 方法 会 在 每 个 测试 结束 后 自动 运行 。 


20.5.1 将 @Unit 用 于 泛 型 
泛 型 为 @Unit 出 了 一 个 难题 ， 因 为 我 们 不 可 能 “泛泛 地 测试 "。 我 们 必须 针对 某 个 特定 类 型 的 
参数 或 参数 集 才能 进行 测试 。 解 决 的 办 法 很 简单 : 让 测试 类 继承 自 泛 型 类 的 一 个 特定 版 本 即 可 。 
下 面 是 一 个 堆栈 的 例子 : 


//: annotations/StackL.java 

// A stack built on a linkedList. 
package annotations; 

import java.util.*; 





public class StackL<T> { 
private LinkedList<T> list = new LinkedList<T>(); 
public void push(T v) { list.addFirst(v); } 
public T top() { return list.getFirst(); } | 
public T pop() { return list.removeFirst(); } 

} Mi~ 


要 测试 String 版 的 堆栈 ， 就 让 测试 类 继承 自 StackL<String>: 


//: annotations/StackLStringTest.java 
// Applying @Unit to generics. 

package annotations; 

import net.mindview.atunit.*; ” 
import net.mindview.util.*; 


public class StackLStringTest extends StackL<String> { 
@Test void _push() { 
push("one™); 
assert top( 
push("two") 
assert top() .equals("two"); 
} 
@Test void _pop() { 
push(“one™) ; 
push("two"); 
assert pop().equals("two"); 


equals ("one"); 
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assert pop().equals("one") ; 


} 
@Test void _top() { 
push( "A: 
push("B"); 
assert top().equals("B"): 
assert top().equats("B"); 





} 
public static void main(String[] args) throws Exception { 
OSExecute. command ( 
"java net.mindview.atunit .AtUnit StackLStringTest") ; 


} 
} /* Output: 
annotations. StackLStringTest 


.to 
OK (3 tests) 
Whew 


这 种 方法 潜在 的 唯一 缺点 是 ， 继承 使 我 们 失去 了 访问 被 测试 的 类 中 的 private 方 法 的 能 力 。 
如 果 这 对 你 很 重要 ， 那 你 要 么 将 private 方 法 改 为 protected， 要 么 添加 一 个 非 private 的 
@TestProperty 方 法 ， 由 它 来 调用 private 方 法 ( 稍 候 我 们 会 看 到 ，AtUnitRemover 工 具 会 将 
@TestProperty 方 法 从 产品 的 代码 中 自动 删除 掉 ) 。 

练习 8，(2) 写 一 个 带 有 private 方 法 的 类 ， 然 后 像 上 介绍 的 那样 添加 一 个 非 private @TestPro- 
perty 方 法 ， 并 在 你 的 测试 代码 中 调用 此 方法 。 

练习 9，(2) 为 HashMap 编 写 一 些 基本 的 @Unit 测 试 。 

练习 10: (2) 从 本 书 中 选择 一 个 示例 程序 ， 为 其 编写 @Unit 测 试 。 r 
20.5.2 不 需要 任何 “套件 ” 

与 JUnit 相 比 ，@Unit 有 一 个 比较 大 的 优点 ， 就 是 @Unit 不 需要 “套件 ”(suites)。 在 JUnit 中 ， 
程序 员 必 须 告诉 测试 工具 你 打算 测试 什么 ， 这 就 要 求 用 套件 来 组 织 测试 ， 以 便 JUnit 能 够 找到 它 
们 ， 并 运行 其 中 包含 的 测试 。 

@Unit 只 是 简单 地 搜索 类 文件 ， 检 查 其 是 否 具有 恰当 的 注解 ， 然 后 运行 @Test 方 法 。 我 的 主 
要 目标 就 是 使 @Unit 测 试 系统 尽 可 能 的 透明 ， 使 得 程序 员 在 用 它 的 时 修 只 需 添加 @Test 方 法 ， 而 
不 需要 像 JUnit 等 其 他 单元 测试 框架 所 要 求 的 那些 特殊 的 编码 或 者 知识 。 不 过 ， 如 果 说 编写 测试 
不 会 遇 到 任何 障碍 ， 这 也 不 太 可 能 ， 因 此 @Unit 会 尽量 让 这 些 困 难 变 得 微不足道 。 希 望 通过 这 
种 方式 ， 程 序 员 会 更 乐意 编写 测试 。 

20.5.3 实现 @Unit 

首先 ， 我 们 需要 定义 所 有 的 注解 类 型 。 这 些 都 是 简单 的 标签 ， 并 且 没 有 属性 。@Test 标 签 在 

本 章 开 头 已 经 定义 过 了 ， 这 里 是 其 他 所 需 的 注解: 


//: net/mindview/atunit/TestObjectCreate.java 
// The @Unit @TestObjectCreate tag. 

package net.mindview.atunit; 

import java.lang.annotation.*; 


@Target (Element Type. METHOD) 
@Retent ion(RetentionPolicy.RUNTIME) 
public @interface TestObjectCreate {} ///:~ 


//:_net/mindview/atunit/TestObjectCleanup. java 
// The @Unit @TestObjectCleanup tag. 

package net.mindview.atunit; 

import java.lang.annotation.*; 


@Target (Element Type. METHOD) 
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@Retent ion (RetentionPol icy RUNTIME) 
public @interface TestObjectCleanup {} ///:~ 


//: net/mindview/atunit/TestProperty. java 
// The @Unit @TestProperty tag. 

package net.mindview.atunit; 

import java. lang. annotation.*; 


// Both fields and methods may be tagged as properties: 
@Target({ElementType.FIELD, ElementType.METHOD}) 
@Retention(RetentionPolicy.RUNTIME) 

public @interface TestProperty {} ///:~ 


所 有 测试 的 保留 属性 必须 是 RUNTIME， 因 为 @Unit 系 统 必须 在 编译 后 的 代码 中 查询 这 些 
注解 。 
要 实现 该 系统 ， 并 运行 测试 ， 我 们 还 需 使 用 反射 机 制 来 抽取 注解 。 下 面 这 个 程序 通过 注解 
中 的 信息 ， 决 定 如 何 构造 测试 对 象 ， 并 在 测试 对 象 上 运行 测试 。 正 是 由 于 注解 的 帮助 ， 这 个 程 
序 才 如 此 短小 而 直接 : 


/1/1: net/mindview/atunit/AtUnit.java 

7/ An annotation-based unit-test framework. 
// {RunByHand) 

package net.mindview.atunit; 

import java. lang.reflect. 

import java.io. 
import java.util.*; 
import net.mindview.util.*; 

import static net.mindview.util.Print.*; 








public class AtUnit implements ProcessFiles.Strategy { 
static Class<?> testClass; 
static List<String> failedTests= new ArrayList<String>(); 
static long testsRun = 0; 
static long failures = 
public static void main(String{] args) throws Exception { 
ClassLoader . getSystemClassLoader() 
.setDefaultAssertionStatus(true); // Enable asserts 
new ProcessFiles(new AtUnit(), "class").start (args); 
if(failures == 6) 
print("OK (" + testsRun + " tests)"); 
else { 
print("(" + testsRun + " tests)"); 
print("\n>>> ”+ failures + " FAILURE" + 
(failures > 1 ? "S" : "") +" <<<"); 
for (String failed : failedTests) . 
print(” " + failed); 





) 
) 
public void process(File cFile) { 
try { 
String cName = ClassNameF inder.thisClass( 
BinaryFile.read(cFile)); 
if(!cNane.contains(".")) 
return; // Ignore unpackaged classes 
testClass = Class. forName(cName) 
} catch(Exception e) { 
throw new RuntimeException(e); 
) 
TestMethods testMethods = new TestMethods(); 
Method creator = null; 
Method cleanup null; 
for(Method m : testClass.getDeclaredMethods()) { 
testMethods.addIfTestMethod(m) ; 11097] 
if(creator == null) 
creator = checkForCreatormethod(m) : 
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if(cleanup == null) 
cleanup = checkForCleanupMethod(m) ; 
} 
if(testMethods.size() > 0) { 
if(creator == null) 
try { s 
if (Modifier. isPublic(testClass 
-getDeclaredConstructor().getNodifiers())) { 
print("Error: " + testClass + 
"default constructor must be public"); 
System.exit(1); 
} 
} catch(NoSuchMethodException e) { ' 
// Synthesized default constructor; OK 
} 


print(testClass.getName()); 
} 
for(Method m : testMethods) { 
printnb(" . * + m.getName() +* *); 
try { 
Object testobject = createTestObject (creator); 
boolean success = false; 
try { 
if (m.getReturnType() equals (boolean.class)) 
Success = (Boolean)m. invoke(testObject) ; 
else { 
m, invoke(testObject); 
success = true; // If no assert fails 
} 
} catch(InvocationTargetException e) { 
// Actual exception is inside e: 
print(e.getCause()): 





print(success ? “" : “(failed)"): 
testsRunt+; 
if(!success) { 
failures++; 
failedTests.add(testClass.getName() + 
": " + m.getName())! 
} + 
if(cteanup != null) 
cleanup. invoke(testObject, testObject); 
} catch(Exception e) { 
throw new Runt imeException(e); 
} 


} } 
Static class TestMethods extends ArrayList<Method> { 
void addIfTestMethod (Method m) { 
if(m.getAnnotation(Test.class) == null) 
return; 
if (1 (m.getReturnType() .equals(boolean.class) || 
m. getReturnType() .equals(void.class))) 
throw new RuntimeException("@Test method” + 
“must return boolean or void"); 
m.setAccessible(true); // In case it's private, etc. 
add (m) ; 
) } 
private static Method checkForCreatorMethod(Method m) { 
if (m.getAnnotation(TestObjectCreate.class) == null) 
return null: 
if(1m.getReturnType() .equals(testClass)) 
throw new RuntimeException("@TestObjectCreate "+ 
"must return instance of Class to be tested"); 
if ((m.getModifiers() & 
java. lang.reflect.Modifier.STATIC) < 1) 
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throw new RuntimeException("@TestObjectCreate ”+ 
“must be static."); 
m.setAccessible(true); 
return m; 


private static Method checkForCleanupMethod(Method m) { 
if (m.getAnnotation(TestObjectCleanup.class) == null) 
return null; 
if (1m. getReturnType() .equals(void.class)) 
throw new RuntimeException("@TestObjectCleanup ”+ 
“must return void"); 
if ((m.getModifiers() & 
java. lang.reflect Modifier.STATIC) < 1) 
throw new RuntimeException("@TestObjectCleanup ”+ 
“must be static."); 
if (m.getParameterTypes().length == 6 || 
m.getParameterTypes(){@] != testClass) 
throw new Runt imeException("@TestObjectCleanup * 








“must take an argument of the tested type."); 1095] 
m.setaccessible(true); 
return m; 


} 
private static Object createTestObject(Hethod creator) { 
if(creator != null) { 
try { 
return creator. invoke(testClass) ; 
} catch(Exception e) { 
throw new RuntimeException("Couldn't run " + 
“@TestObject (creator) method."); 


} 
} else { // Use the default constructor: 
try { 
return testClass.newInstance(); 
} catch(Exception e) { 
throw new RuntimeException(*Couldn't create a * + 
“test object. Try using a @TestObject method."): 
} 


} 

} ys 

AtUnit.java 使 用 了 net.mindview.util 中 的 ProcessFiles 工 具 。 这 个 类 还 实现 了 ProcessFiles 
,Strategy 接口 ， 该 接口 包含 process0 方 法 。 如 此 一 来 ， 便 可 以 将 一 个 AtUnit 实 例 传 给 ProcessFiles 
的 构造 路 。ProcessFiles 构 造 器 的 第 二 个 参数 告诉 ProcessFiles 查 找 所 有 扩展 名 为 class 的 文件 。 

如 果 你 没有 提供 命令 行 参 数 ， 这 个 程序 会 遍历 当前 目录 。 你 也 可 以 为 其 提供 多 个 参数 ， 可 
以 是 类 文件 〈 带 有 或 不 带 .class 扩 展 名 都 可 ) ， 或 者 是 一 些 目录 。 由 于 @Unit 将 会 自动 找到 可 测试 
的 类 和 方法 ， 所 以 没有 “套件 ”机 制 的 必要 。 。 

AtUnitjava 必 须要 解决 一 个 问题 ， 就 是 当 它 找到 类 文件 时 ， 实 际 引 用 的 类 名 (含有 包 ) 并 
非 一 定 就 是 类 文件 的 名 字 。 为 了 从 中 解读 信息 ， 我 们 必须 分 析 该 类 文件 ， 这 很 重要 ， 因 为 这 种 
名 字 不 一 致 的 情况 确实 可 能 出 现 。 所 以 ， 当 找到 一 个 .class 文件 时 ， 第 一 件 事情 就 是 打开 该 文 蕊 0 
件 ， 读 取 其 二 进 制 数据 ， 然 后 将 其 交 给 ClassNameFinderthisClass0。 从 这 里 开始 ， 我 们 将 进入 
“ 字 节 码 工程 ”的 领域 ， 因 为 我 们 实际 上 是 在 分 析 一 个 类 文件 的 内 容 : 


/1/: net/mindview/atunit/ClassNameFinder .java 
package net.mindview.atunit; 








O 现在 还 不 清楚 为 何 测 试 所 属 的 类 的 默认 构造 嚣 必须 是 public， 如 果 不 是 的 话 ， 调 用 newInstantce0 方 法 会 导致 
程序 中 止 ( 没 有 异常 抛 出 )。 
© Jeremy Meyer 与 我 在 这 个 问题 上 花 了 一 整 天 。 
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‘import java.io.*; 
import java.uti 
import net.mindview.util.*; 

import static net.mindview.util.Print.*; 





public class ClassNameFinder { 
public static String thisClass(byte(] classBytes) { 
Map<Integer ,Integer> offsetTable = 
new HashMap<Integer , Integer>() ; 
Map<Integer „String> classNameTable = 
new HashMap<Integer.String>(): 
try { 
DataInputStream data = new DataInputStream( 
new ByteArrayInputStream(classBytes) ); 
int magic = data.readInt(); // @xcafebabe 
int minorVersion = data.readShort(): 
int majorVersion = data.readShort( 
int constant_pool_count = data.readShort(); 
int[] constant_pool = new int{constant_pool_count] ; 
for(int i = 1; i < constant_pool_count; i++) { 
int tag = data.read(); 
int tableSize; 
switch(tag) { 
case 1: // UTF 
int length = data.readShort(); 
char[] bytes = new char [length] ; 
for(int k = @; k < bytes.length; k++) 
bytes[k] = (char)data.read(); 
String className = new String(bytes) ; 
classNameTable.put(i, className) ; 
break; 
case 5: // LONG 
case 6: // DOUBLE 
data.readLong(): // discard 8 bytes 
i++; // Special skip necessary 
break; 
case 7: // CLASS 
int offset = data.readShort(): 
offsetTable.put(i, offset); 
break; 
case 8: // STRING 
data.readShort(); // discard 2 bytes 
break; 
case 3: // INTEGER 
case 4: // FLOAT 
case 9: // FIELD_REF 
case 10: // METHOD_REF 
case 11: // INTERFACE_METHOD_REF 
case 12: // NAME_AND_TYPE 
data.readint(); // discard 4 bytes; 
break; 
default: 
throw new RuntimeException("Bad tag ”+ tag 
和 


} 
short access_flags = data.readShort(); 
int this_class = data.readShort(); 
int super_class = data.readShort(); 
return classNameTable.get( 
offsetTable.get(this_class)).replace('/', '. 
} catch(Exception e) { 
throw new Runt imeException(e); 
} 
} 
// Demonstration: 
public static void main(String{] args) throws Exception { 
if(args.length > 6) { 
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for(String arg : args) 
print(thisClass(BinaryFile.read(new File(arg)))): 
, ay walk the entire tree: 
for(File klass : Directory.walk(".", “.*\\.class")) 
print(thisClass(BinaryFile.read(klass))); 

} Vis 

虽然 无 法 在 这 里 介绍 其 中 所 有 的 细节 ， 但 每 个 类 文件 都 必须 遵循 一 定 的 格式 ， 而 我 已 经 尽 
量 用 有 意义 的 域名 字 来 表示 这 些 从 ByteArrayInputStream 中 提出 取 来 的 数据 片断 。 通 过 施加 在 
输入 流 上 的 读 操作 ， 你 能 看 出 每 个 信息 片 的 大 小 。 例 如 ， 每 个 类 文件 的 头 32 个 bit 总 是 一 个 “ 神 
秘 的 数字 ”hex0xcafebabe。 ， 而 接 下 来 的 两 个 short 值 是 版 本 信息 。 常 量 池 包 含 了 程序 中 的 常量 ， 
所 以 这 是 一 个 可 变 的 值 。 接 下 来 的 short 告 诉 我 们 这 个 常量 地 有 多 大 ， 然 后 我 们 为 其 创建 一 个 尺 
寸 合适 的 数组 。 常 县 池 中 的 每 一 个 元 素 ， 其 长 度 可 能 是 一 个 固定 的 值 ， 也 可 能 是 可 变 的 值 ， 因 
此 我 们 必须 检查 每 一 个 常量 起 始 的 标记 ， 然 后 才能 知道 该 怎么 做 ， 这 就 是 switch 语 句 中 的 工作 。 
我 们 并 不 打算 精确 地 分 析 类 中 的 所 有 数据 ， 仅 仅 是 从 文件 的 起 始 一 步 一 步 地 走 ， 直 到 取得 我 们 
所 需 的 信息 ， 因 此 你 会 发 现 ， 在 这 个 过 程 中 我 们 丢弃 了 大 量 的 数据 。 关 于 类 的 信息 都 保存 在 
classNameTable 和 offsetTable 中 。 在 读 完了 常量 池 之 后 ， 就 找到 了 this_class 信 息 ， 这 是 
offsetTable 中 的 一 个 坐标 ， 通 过 它 能 够 找到 一 个 进入 classNameTable 的 坐标 ， 然 后 就 可 以 得 到 我 
们 所 需 的 类 的 名 字 了 。 

现在 ,让 我 们 回 到 AtUnit.java 程 序 ，process0 方 法 现在 拥有 了 类 的 名 字 ， 然 后 检查 它 是 否 
包含 “."， 如 果 有 就 表示 该 类 定义 于 一 个 包 中 。 没 有 包 的 类 将 被 忽略 。 如 果 一 个 类 在 包 中 ， 那 么 
我 们 就 可 以 使 用 标准 的 类 加 载 器 并 通过 Class.forName0 将 其 加 载 进来 。 现 在 ， 我 们 终于 可 以 开 
始 对 这 个 类 进行 @Unit 注 解 的 分 析 工 作 了 。 

我 们 只 需 关 心 三 件 事情 : 首先 是 @Test 方 法 ， 它 们 将 被 保存 在 TestMethos 列 表 中 ， 然 后 检查 
是 否 具有 @TestObjectCreate 和 @TestObjectCleanup 方 法 。 从 代码 中 可 以 看 到 ， 我 们 通过 调用 相 
应 的 方法 来 查询 注解 从 而 找到 这 些 方法 。 

每 当 找到 一 个 @Test 方 法 ， 就 打印 出 当前 的 类 的 名 字 ， 于 是 观察 者 立刻 就 可 以 知道 发 生 了 什 
么 。 接 下 来 开始 执行 测试 ， 也 就 是 打印 出 方法 名 ， 然 后 调用 createTestObjectO (如 果 存 在 一 个 
加 了 @TestObjeetCreate 注 解 的 方法 ) ， 或 者 调用 默认 的 构造 器 。 一 旦 创建 出 测试 对 象 ， 就 调用 
其 上 的 测试 方法 。 如 果 测试 返回 一 个 boolean 值 ， 就 捕获 该 结果 。 如 果 测 试 方法 没有 返回 值 ， 那 
么 当 没有 异常 发 生 时 ， 我 们 就 假设 测试 成 功 ， 反 之 ， 如 果 当 assert 失 败 或 有 任何 异常 抛 出 时 ,就 
说 明 测试 失败 ， 这 时 将 异常 信息 打印 出 来 以 显示 错误 的 原因 。 如 果 有 失败 的 测试 发 生 ， 那 么 还 
要 统计 失败 的 次 数 ， 并 将 失败 的 测试 所 属 的 类 和 方法 的 名 字 加 入 failedTests， 以 便 最 后 将 其 报告 
给 用 户 。 

练习 11: (5) 向 @Unit 中 加 入 一 个 @TestNote 注 解 ， 以 便 这 些 附加 的 信息 在 测试 时 能 够 显示 
出 来 。 

20.5.4 移 除 测试 代码 

对 许多 项 目 而 言 ， 在 发 布 的 代码 中 是 否 保留 测试 代码 并 没什么 区 别 (特别 是 在 如 果 你 将 所 
有 的 测试 方法 都 声明 为 private 的 情况 下 ， 如 果 你 喜欢 就 可 以 这 么 做 )， 但 是 在 有 的 情况 下 ， 我 们 
确实 希望 将 测试 代码 清除 掉 ， 精 简 发 布 的 程序 ， 或 者 就 是 不 希望 测试 代码 暴露 给 客户 。 


O 关于 这 个 数字 有 许多 传说 ， 不 过 考虑 到 Java 是 由 书 呆 子 创造 出 来 的 ， 我 们 可 以 做 一 个 合理 的 猜测 ， 他 可 能 正 幻 
想 着 咖啡 店 中 的 某 个 女人 。 
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与 自己 动手 删除 测试 代码 相 比 ， 这 需要 更 复杂 的 字 节 码 工程 。 不 过 开源 的 Javassist 工 具 类 
库 ? 将 字 节 码 工程 带 入 了 一 个 可 行 的 领域 。 下 面 的 程序 接受 一 个 -r 标 志 作为 其 第 一 个 参数 ， 如 果 
你 提供 了 该 标志 ， 那 么 它 就 会 删除 所 有 的 @Test 注 解 ， 如 果 你 没有 提供 该 标记 ， 那 它 则 只 会 打印 
出 @Test 注 解 。 这 里 同样 使 用 ProcessFiles 来 遍历 你 选择 的 文件 和 目录 ; 


//: net/mindview/atunit/AtUnitRemover.java 
// Displays @Unit annotations in compiled class files. If 
// first argument is "-r", @Unit annotations are removed. 
44 (args: ..} 
// (Requires: javassist.bytecode.ClassFile; 
// You must install the Javassist library from 
// http://sourceforge.net/projects/jboss/ } 
package net .mindview.atunit; 
import javassist.*; 
import javassist.expr.* 
import javassist.bytecode.*; 
import javassist. bytecode. annotation. *; 
import java.io.*; 
import java.util. 
import net.mindview.util.*; 
import static net.mindview.util.Print. 
public class AtUnitRemover 
implements ProcessFiles. Strategy ( 
private static boolean remove = false; 
public static void main(String{] args) throws Exception { 
if(args.length > © && args[9] .equals("-r")) { 
remove = true; 
String[] nargs = new Stringlargs.length - 1]; 
System. arraycopy(args, 1, nargs, ©, nargs. length); 
args = nargs; 
} 
new ProcessFiles( 
new AtUnitRemover(), "class").start (args); 
} 


public void process(File cFile) { 
boolean modified = false; 
try { 
String cName = ClassNameF inder.thisClass( 
BinaryFile.read(cFile)): + 
if (!cName.contains(".")) 
return; // Ignore unpackaged classes 
ClassPool cPool = ClassPool.getDefault(); 
CtClass ctClass = cPool.get (cName) ; 
for (CtMethod method : ctClass.getDeclaredMethods()) { 
MethodInfo mi = method. getMethodInfo() ; 
AnnotationsAttribute attr = (AnnotationsAttribute) 
mi .getAttr ibute(Annotationsattribute.visibleTag) ; 
if(attr == null) continue; 
for (Annotation ann : attr.getAnnotations()) { 
if (ann. getTypeName() 
.startsWith("net.mindview.atunit")) { 
print(ctClass.getName() + " Method: * 
+ mi.getName() +" * + ann); 
if(remove) { 
ctClass.removeMethod (method) ; 
modified = true; 
} 
} 

















} 
} 
// Fields are not removed in this version (see text). 
if (modified) 


© 感谢 Shigeru Chiba 博 士 创建 了 该 工具 ， 以 及 他 对 我 开发 AtUnitRemoverjava 的 帮助 。 








ctClass. toBytecode(new DataQutputstream( 
new FileQutputStream(cFite))); 
ctClass.detach(); 
} catch(Exception e) { 
throw new RuntimeException(e): 


} 

dM ~~ 

ClassPool 是 一 种 全 景 ， 它 记录 了 你 正在 修改 的 系统 中 的 所 有 的 类 ， 并 能 够 保证 所 有 类 在 修 
改 后 的 一 致 性 。 你 必须 从 ClassPool 中 取得 每 个 CtClass， 这 与 使 用 类 加 载 器 和 Class.forNameO 向 
JVM 加 载 类 的 方式 类 似 。 

CtClass 包 含 的 是 类 对 象 的 字 节 码 ， 你 可 以 通过 它 取得 类 有 关 的 信息 ， 并 且 操 作 类 中 的 代码 。 
在 这 里 ， 我 们 调用 getDeclaredMethods0 (与 Java 的 反射 机 制 一 样 )， 然 后 从 每 个 CtMethod 对 象 
中 取得 一 个 MethodInfo 对 象 。 通 过 该 对 象 ， 我 们 察看 其 中 的 注解 信息 。 如 果 一 个 方法 带 有 
net.mindview.atunit 包 中 的 注解 ， 就 将 该 方法 删除 掉 。 

如 果 类 被 修改 过 了 ， 就 用 新 的 类 覆盖 原始 的 类 文件 。 

在 撰写 本 书 时 ，Javassist 刚 刚 加 入 了 “删除 ”功能 ， 同 时 我 们 发 现 ， 删 除 @TestProperty 
域 比 删除 方法 复杂 得 多 。 因 为 ， 有 些 静 态 初始 化 的 操作 可 能 会 引用 这 些 域 ， 所 以 你 不 能 简单 地 
将 其 删除 。 因 此 AtUnitRemover 的 当前 版 本 只 删除 @Unit 方 法 。 不 过 ， 你 应 该 查看 一 下 Javassist 
网 站 的 更 新 ， 因 为 删除 域 的 功能 以 后 可 能 也 将 实现 。 与 此 同时 ， 对 于 AtUnitExternalTextjava 演 
示 的 外 部 测试 方法 ， 可 以 直接 删除 测试 代码 生成 的 类 文件 ， 从 而 到 达 删 除 所 有 测试 的 目的 。 


20.6 总 结 


注解 是 Java 引 入 的 一 项 非常 受 欢迎 的 补充 。 它 提供 了 一 种 结构 化 的 ， 并 且 具 有 类 型 检查 能 
力 的 新 途径 ， 从 而 使 得 程序 员 能 够 为 代码 加 入 元 数据 ， 而 不 会 导致 代码 杂乱 且 难 以 阅读 。 使 用 
注解 能 够 帮助 我 们 避免 编写 累 装 的 部 署 描述 文件 ， 以 及 其 他 生成 的 文件 。 而 Javadoc 中 的 
@deprecated 被 @Deprecated 注 解 取代 的 事实 也 说 明 ， 与 注释 性 文字 相 比 ， 注 解 绝 对 更 适合 用 于 
描述 类 相关 的 信息 。 

Java SE5 仅 提供 了 很 少 的 内 置 注 解 。 这 意味 着 如 果 你 在 别处 找 不 到 可 用 的 类 库 ， 那 就 只 能 自 
己 创 建新 的 注解 以 及 相应 的 处 理 器 。 有 了 apt 工 具 的 帮助 ， 程 序 员 可 以 同时 编译 新 产生 的 源 文件 ， 
以 及 简化 构建 过 程 ， 不 过 就 目前 的 情况 看 ，mirror API 只 能 给 予 你 一 些 基本 功能 ， 帮 助 你 找到 
Java 类 定义 中 的 元 素 。 正 如 你 已 经 看 到 的 ，Javassist 能 够 用 来 操作 字 节 码 ， 或 者 你 也 可 以 编写 自 
已 的 字 节 码 操作 工具 。 

这 个 状况 将 来 一 定 会 改善 ，API 提 供 方 ， 以 及 各 种 framework 一 定 会 将 注解 包含 在 其 提供 的 
工具 集 内 。 通 过 @Unit 系 统 ， 我 们 可 以 想像 得 到 ， 注 解 很 可 能 将 引发 Java 编 程 体验 的 巨大 改变 。 

所 选 习题 的 答案 都 可 以 在 名 为 The Thinking in Java Annotated Solution Guide 的 电子 文档 中 找 
到 ， 读 者 可 以 从 www.MindView.net 购 买 此 文档 。 





© ERMER, Shigeru Chiba 博 士 将 CtClass.removeMethodQ 加 入 了 其 中 。 
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第 21 章 并 发 


到 目前 为 止 ， 你 学 到 的 都 是 有 关 顺 序 编程 的 知识 。 即 程序 中 的 所 有 事物 在 任意 时 刻 都 只 能 
执行 一 个 步骤 。 

编程 问题 中 相当 大 的 一 部 分 都 可 以 通过 使 用 顺序 编程 来 解决 。 然 而 ， 对 于 某 些 问题 ， 如 果 
能 够 并 行 地 执行 程序 中 的 多 个 部 分 ， 则 会 变 得 非常 方便 甚至 非常 必要 ， 因 为 这 些 部 分 要 么 看 起 
来 在 并 发 地 执行 ， 要 么 在 多 处 理 器 环境 下 可 以 同时 执行 。 

并 行 编程 可 以 使 程序 执行 速度 得 到 极 大 提高 ， 或 者 为 设计 某 些 类 型 的 程序 提供 更 易 用 的 模 
N, 或 者 两 者 皆 有 。 但 是 ， 熟 练 掌握 并 发 编程 理论 和 技术 ， 对 于 到 目前 为 止 你 在 本 书 中 学 习 到 
的 所 有 知识 而 言 ， 是 一 种 飞跃 ， 并 且 是 通 向 高 级 主题 的 中 介 。 本 章 只 能 作为 一 个 介绍 ， 即 便 融 
会 贯通 了 本 章 的 内 容 ， 也 绝 不 意味 着 你 就 是 一 个 优秀 的 并 发 程序 员 了 。 

正如 你 应 该 看 到 的 ， 当 并 行 执行 的 任务 彼此 开始 产生 互相 干涉 时 ， 实 际 的 并 发 问题 就 会 接 
是 而 至 。 这 可 能 会 以 一 种 微妙 而 偶然 的 方式 发 生 ， 我 们 可 以 很 公正 地 说 ， 并 发 “具有 可 论证 的 
确定 性 ， 但 是 实际 上 具有 不 可 确定 性 "。 这 就 是 说 ， 你 可 以 得 出 结论 ， 通 过 仔细 设计 和 代码 审查 ， 
编写 能 够 正确 工作 的 并 发 程序 是 可 能 的 。 但 是 ， 在 实际 情况 中 ， 更 容易 发 生 的 情况 是 所 编写 的 
并 发 程序 在 给 定 适当 条 件 的 时 候 ， 将 会 工作 失败 。 这 些 条 件 可 能 从 来 都 不 会 实际 发 生 ， 或 者 发 
生得 不 是 很 频繁 ， 以 至 于 在 测试 过 程 中 不 会 磁 上 它们 。 实 际 上 ， 你 可 能 无 法 编写 出 能 够 针对 你 
的 并 发 程序 生成 故障 条 件 的 测试 代码 。 所 产生 的 故障 经 常 是 偶尔 发 生 的 ， 并 且 经 常 是 以 客户 抱 
怨 的 形式 出 现 的 。 这 是 研究 并 发 问题 的 最 强 理由 : 如 果 视 而 不 见 ， 你 就 会 遭 其 反 叭 。 

因此 ， 并 发 看 起 来 充满 了 危险 ， 如 果 你 对 它 有 些 恨 惧 ， 这 可 能 是 件 好 事 。 尽 管 Java SE5 在 并 
发 方面 做 出 了 显著 的 改进 ， 但 是 仍旧 没有 像 编 译 期 验证 或 检查 型 异常 这 样 的 安全 网 ， 在 你 犯错 
误 的 时 候 告 知 你 。 使 用 并 发 时 ， 你 得 自食其力 ， 并 且 只 有 变 得 多 疑 而 自信 ， 才 能 用 Java 编 写 出 
可 靠 的 多 线程 代码 。 

有 时 人 们 会 认为 并 发 对 于 介绍 语言 的 书 来 说 太 高 级 了 ， 因 此 不 适合 放 在 其 中 。 他 们 认为 并 
发 是 一 个 独立 主题 ， 可 以 单独 来 处 理 ， 并 且 对 于 少数 出 现在 日 常 的 程序 设计 中 的 情况 (例如 图 
形 化 用 户 界面 )， 可 以 用 特殊 的 惯用 法 来 处 理 。 如 果 你 可 以 回避 ， 为 什么 还 要 介绍 这 么 复杂 的 主 
题 呢 ? 

唉 ， 如 果 是 这 样 就 好 了 。 遗 憾 的 是 ， 你 无 法 选择 何 时 在 你 的 Java 程 序 中 出 现 线程 。 仅 仅 是 
你 自己 没有 启动 线程 并 不 代表 你 就 可 以 回避 编写 使 用 线程 的 代码 。 例 如 ，Web 系 统 是 最 常见 的 
Java 应 用 系统 之 一 ,而 基本 的 Web 库 类 、Servlet 具 有 天 生 的 多 线程 性 一 一 这 很 重要 ， 因 为 Web 服 
务 器 经 常 包含 多 个 处 理 器 ， 而 并 发 是 充分 利用 这 些 处 理 器 的 理想 方式 。 即 便 是 像 Servlet 这 样 看 
起 来 很 简单 的 情况 ， 你 也 必须 理解 并 发 问题 ， 从 而 能 正确 地 使 用 它们 。 图 形 化 用 户 界面 也 是 类 
似 的 情况 ， 你 将 在 第 22 章 中 看 到 。 尽 管 Swing 和 SWT 类 库 都 拥有 针对 线程 安全 的 机 制 ， 但 是 不 理 
解 并 发 ， 就 很 难 了 解 如 何 正确 地 使 用 它们 。 

Java 是 一 种 多 线程 语言 ， 并 且 提 出 了 并 发 问题 ， 不 管 你 是 否 意识 到 了 。 因 此 ， 有 很 多 使 用 
中 的 Java 程 序 ， 要 么 只 是 偶尔 工作 ， 要 么 在 大 多 数 时 间 里 工作 ， 并 且 会 由 于 未 发 现 的 并 发 缺陷 
而 时 不 时 地 神秘 崩溃 。 有 时 这 种 崩溃 是 温和 的 ， 但 有 时 却 意 味 着 重要 数据 的 丢失 ， 并 且 如 果 没 
有 意识 到 并 发 问题 ， 你 可 能 最 终 会 认为 问题 出 在 其 他 什么 地 方 ， 而 不 在 你 的 软件 中 。 如 果 程 序 
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被 迁移 到 多 处 理 器 系统 中 ， 这 些 种 类 的 问题 还 会 被 暴露 或 放大 。 基 本 上 ， 了 解 并 发 可 以 使 你 意 
识 到 明显 正确 的 程序 可 能 会 展示 出 不 正确 的 行为 。 

学 习 并 发 编程 就 像 进入 了 一 个 全 新 的 领域 ， 有 点 类 似 于 学 习 一 门 新 的 编程 语言 ， 或 者 至 少 
是 学 习 一 整套 新 的 语言 概念 。 要 理解 并 发 编程 ， 其 难度 与 理解 面向 对 象 编程 差不多 。 如 果 你 花 
点 儿 工 夫 ， 就 能 明白 其 基本 机 制 ， 但 要 想 真正 地 掌握 它 的 实质 ， 就 需要 深入 的 学 习 和 理解 。 本 
章 的 目标 就 是 要 让 读者 对 并 发 的 基本 知识 打下 坚实 的 基础 ， 从 而 能 够 理解 其 概念 并 编写 出 合理 
的 多 线程 程序 。 注 意 ， 你 可 能 很 容易 就 会 变 得 过 分 自信 ， 在 你 编写 任何 复杂 程序 之 前 ， 应 该 学 
习 一 下 专门 讨论 这 个 主题 的 书籍 。 


21.1 并 发 的 多 面 性 


并 发 编程 令 人 困惑 的 一 个 主要 原因 是 : 使 用 并 发 时 需要 解决 的 问题 有 多 个 ， 而 实现 并 发 的 
方式 也 有 多 种 ， 并 且 在 这 两 者 之 间 没有 明显 的 映射 关系 (而 且 通 常 只 具有 模糊 的 界线 )。 因 此 ， 
你 必须 理解 所 有 这 些 问题 和 特例 ， 以 便 有 效 地 使 用 并 发 。 

用 并 发 解决 的 问题 大 体 上 可 以 分 为 “速度 ”和 “设计 可 管理 性 ”两 种 。 

21.1.1 更 快 的 执行 

速度 问题 初 听 起 来 很 简单 : 如 果 你 想 要 一 个 程序 运行 得 更 快 ， 那 么 可 以 将 其 断 开 为 多 个 片 
段 ， 在 单独 的 处 理 器 上 运行 每 个 片段 。 并 发 是 用 于 多 处 理 器 编程 的 基本 工具 。 当 前 ，Moore 定 律 
已 经 有 些 过 时 了 (至 少 对 于 传统 芯片 是 这 样 )， 速 度 提高 是 以 多 核 处 理 器 的 形式 而 不 是 更 快 的 芯 
片 的 形式 出 现 的 。 为 了 使 程序 运行 得 更 快 ， 你 必须 学 习 如 何 利用 这 些 额外 的 处 理 器 ， 而 这 正 是 
并 发 赋予 你 的 能 力 。 

如 果 你 有 一 台 多 处 理 器 的 机 器 ， 那 么 就 可 以 在 这 些 处 理 器 之 间 分 布 多 个 任务 ， 从 而 可 以 极 
大 地 提高 吞吐 量 。 这 是 使 用 强 有 力 的 多 处 理 器 Web 服 务 器 的 常见 情况 ， 在 为 每 个 请 求 分 配 一 个 
线程 的 程序 中 ， 它 可 以 将 大 量 的 用 户 请 求 分 布 到 多 个 CPU 上 。 

但 是 ， 并 发 通常 是 提高 运行 在 单 处 理 器 上 的 程序 的 性 能 。 

这 听 起 来 有 些 违背 直觉 。 如 果 你 仔细 考虑 一 下 就 会 发 现 ， 在 单 处 理 器 上 运行 的 并 发 程序 开 
销 确实 应 该 比 该 程序 的 所 有 部 分 都 顺序 执行 的 开销 大 ， 因 为 其 中 增加 了 所 谓 上 下 文 切 换 的 代价 
(从 一 个 任务 切换 到 另 一 个 任务 )。 表 面 上 看 ， 将 程序 的 所 有 部 分 当 作 单个 的 任务 运行 好 像 是 开 
销 更 小 一 点 ， 并 且 可 以 节省 上 下 文 切 换 的 代价 。 

使 这 个 问题 变 得 有 些 不 同 的 是 阻塞。 如 果 程 序 中 的 某 个 任务 因为 该 程序 控制 范围 之 外 的 某 
些 条 件 (通常 是 O) 而 导致 不 能 继续 执行 ， 那 么 我 们 就 说 这 个 任务 或 线程 阻塞 了 。 如 果 没 有 并 
发 ， 则 整个 程序 都 将 停止 下 来 ， 直 至 外 部 条 件 发 生变 化 。 但 是 ， 如 果 使 用 并 发 来 编写 程序 ， 那 
么 当 一 个 任务 阻塞 时 ， 程 序 中 的 其 他 任务 还 可 以 继续 执行 ， 因 此 这 个 程序 可 以 保持 继续 向 前 执 
行 。 事 实 上 ， 从 性 能 的 角度 看 ， 如 果 没 有 任务 会 阻塞 ， 那 么 在 单 处 理 器 机 器 上 使 用 并 发 就 没有 
任何 意义 。 

在 单 处 理 器 系统 中 的 性 能 提高 的 常见 示例 是 事件 驱动 的 编程 。 实 际 上 ， 使 用 并 发 最 吸引 人 
的 一 个 原因 就 是 要 产生 具有 可 响应 的 用 户 界面 。 考 虑 这 样 一 个 程序 ， 它 因为 将 执行 某 些 长 期 运 
行 的 操作 ， 所 以 最 终 用 户 输入 会 被 忽略 ， 从 而 成 为 不 可 响应 的 程序 。 如 果 有 一 个 “退出 ”按钮 ， 
那么 你 肯定 不 想 在 你 写 的 每 一 段 代码 中 都 检查 它 的 状态 。 这 会 产生 非常 尴 认 的 代码 ， 而 我 
们 也 无 法 保证 程序 员 不 会 忘记 这 种 检查 。 如 果 不 使 用 并 发 ， 则 产生 可 响应 用 户 界面 的 唯一 方式 
就 是 所 有 的 任务 都 周期 性 地 检查 用 户 输入 。 通 过 创建 单独 的 执行 线程 来 响应 用 户 的 输入 ， 即 使 
这 个 线程 在 大 多 数 时 间 里 都 是 阻塞 的 ， 但 是 程序 可 以 保证 具有 一 定 程度 的 可 响应 性 。 
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程序 需要 连续 执行 它 的 操作 ， 并 且 同 时 需要 返回 对 用 户 界面 的 控制 ， 以 便 使 程序 可 以 响应 
用 户 。 但 是 传统 的 方法 在 连续 执行 其 操作 的 同时 ， 返 回 对 程序 其 余部 分 的 控制 。 事实 上 ， 这 听 
起 来 就 像 是 不 可 能 之 事 ， 好 像 CPU 必 须 同时 位 于 两 处 一 样 ， 但 是 这 完全 是 并 发 造成 的 一 种 错觉 
(在 多 处 理 器 系统 中 ， 这 就 不 只 是 一 种 幻觉 了 )。 

实现 并 发 最 直接 的 方式 是 在 操作 系统 级 别 使 用 进程 。 进 程 是 运行 在 它 自 己 的 地 址 空间 内 的 
自 包容 的 程序 。 多 任务 操作 系统 可 以 通过 周期 性 地 将 CPU 从 一 个 进程 切换 到 另 一 个 进程 ， 来 实 
现 同时 运行 多 个 进程 (程序 )， 尽 管 这 使 得 每 个 进程 看 起 来 在 其 执行 过 程 中 都 是 哆 吹 停 停 。 进 程 
总 是 很 吸引 人 ， 因 为 操作 系统 通常 会 将 进程 互相 隔离 开 ， 因 此 它们 不 会 彼此 干涉 ， 这 使 得 用 进 
程 编程 相对 容易 一 些 。 与 此 相反 的 是 ， 像 Java 所 使 用 的 这 种 并 发 系统 会 共享 诸如 内 存 和 IO 这 样 
的 资源 ， 因 此 编写 多 线程 程序 最 基本 的 困难 在 于 在 协调 不 同 线程 驱动 的 任务 之 间 对 这 些 资源 的 
使 用 ， 以 使 得 这 些 资源 不 会 同时 被 多 个 任务 访问 。 

这 里 有 一 个 利用 操作 系统 进程 的 简单 示例 。 在 编写 本 书 时 ， 我 会 有 规律 地 创建 本 书 当前 状 
态 的 多 个 宛 余 备 份 副本 。 我 会 在 本 地 目录 中 保存 一 个 副本 ， 在 记忆 棒 上 保存 一 个 副本 ， 在 Zip 盘 
上 保存 一 个 副本 ， 还 会 在 远程 FTP 站 点 上 保存 一 个 副本 。 为 了 自动 化 这 个 过 程 ， 我 还 编写 了 一 个 
小 程序 (用 Python 写 的 ， 但 是 其 概念 是 相同 的 )， 它 会 把 本 书 压缩 成 一 个 文件 ， 其 文件 名 中 带 有 
版 本 号 ， 然 后 执行 复制 操作 。 最 初 ， 我 会 顺序 执行 所 有 的 复制 操作 ， 在 启动 下 一 个 复制 操作 之 
前 先 等 待 前 一 个 操作 的 完成 。 但 随后 我 意识 到 ， 每 个 复制 操作 会 依存 储 介质 MO 速度 的 不 同 而 花 
费 不 同 的 时 间 。 既 然 我 在 使 用 多 任务 操作 系统 ， 那 就 可 以 将 每 个 复制 操作 当 作 单 独 的 进程 来 启 
动 ， 并 让 它们 并 行 地 运行 ， 这 样 可 以 加 速 整 个 程序 的 执行 速度 。 当 一 个 进程 受阻 时 ， 另 一 个 进 
程 可 以 继续 向 前 运行 。 

这 是 并 发 的 理想 示例 。 每 个 任务 都 作为 进程 在 其 自己 的 地 址 空间 中 执行 ， 因 此 任务 之 间 根 
本 不 可 能 互相 干涉 。 更 重要 的 是 ， 对 进程 来 说 ， 它 们 之 间 没有 任何 彼此 通信 的 需要 ， 因 为 它们 
都 是 完全 独立 的 。 操 作 系统 会 处 理 确保 文件 正确 复制 的 所 有 细节 ， 因 此 ， 不 会 有 任何 风险 ， 你 
可 以 获得 更 快 的 程序 ， 并 且 完 全 免费 。 

有 些 人 走 得 更 远 ， 提 倡 将 进程 作为 唯一 合理 的 并 发 方式 。， 但 造 憾 的 是 ， 对 进程 通常 会 有 
数量 和 开销 的 限制 ， 以 避免 它 们 在 不 同 的 并 发 系统 之 间 的 可 应 用 性 。 

某 些 编程 语言 被 设计 为 可 以 将 并 发 任务 彼此 隔离 ， 这 些 语言 通常 被 称 为 函数 型 语言 ， 其 中 
每 个 函数 调用 都 不 会 产生 任何 副作用 (并 因此 而 不 能 干涉 其 他 函数 ) ， 并 因此 可 以 当 作 独 立 的 任 
务 来 驱动 。Erlang 就 是 这 样 的 语言 ， 它 包含 针对 任务 之 间 彼 此 通信 的 安全 机 制 。 如 果 你 发 现 程序 
中 某 个 部 分 必须 大 量 使 用 并 发 ， 并 且 你 在 试图 构建 这 个 部 分 时 磁 到 了 过 多 的 问题 ， 那 么 你 可 以 
考虑 使 用 像 Erlang 这 类 专门 的 并 发 语言 来 创建 这 个 部 分 。 

Java 采 取 了 更 加 传统 的 方式 ， 在 顺序 型 语言 的 基础 上 提供 对 线程 的 支持 9 。 与 在 多 任务 操作 
系统 中 分 又 外 部 进程 不 同 ， 线 程 机 制 是 在 由 执行 程序 表示 的 单一 进程 中 创建 任务 。 这 种 方式 产 
生 的 一 个 好 处 是 操作 系统 的 透明 性 ， 这 对 Java 而 言 ， 是 一 个 重要 的 设计 目标 。 例 如 ， 在 OSX 之 
前 的 Macintosh 操 作 系统 版 本 (Java 第 一 个 版 本 的 一 个 非常 重要 的 目标 系统 ) 不 支持 多 任务 ， 因 
此 ， 除 非 在 Java 中 添加 多 线程 机 制 ， 否 则 任何 并 发 的 Java 程 序 都 无 法 移植 到 Macintosh 和 类 似 的 


平台 之 上 ， 这样 就 会 打破 “编写 一 次 ， 到 处 运行 ”的 要 求 ?。 


日 例如 ，Eric Raymond 在 《The Art of UNIX Programming) (Addison-Wesley, 2004) 中 提出 了 这 种 极端 情况 。 

O 可 能 有 人 会 有 异议 ， 认 为 将 并 发 绑 定 到 顺序 型 语言 上 是 一 种 精 粒 的 方式 ， 但 是 你 必须 得 出 自己 的 结论 。 

© 这 个 要 求 从 来 都 没有 完全 实现 过 ，Sun 也 不 再 大 大 吹捧 了 。 具 有 讽刺 意味 的 是 , “编写 一 次 ， 到 处 运行 "并 不 
能 完全 工作 的 原因 也 许 是 因为 多 线程 系统 中 的 问题 而 导致 的 一 一 这 在 Java SE5 中 可 能 已 经 修复 了 。 
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211.2 改进 代码 设计 

在 单 CPU 机 器 上 使 用 多 任务 的 程序 在 任意 时 刻 仍 旧 只 在 执行 一 项 工作 ， 因 此 从 理论 上 讲 ， 
肯定 可 以 不 用 任何 任务 而 编写 出 相同 的 程序 。 但 是 ， 并 发 提供 了 一 个 重要 的 组 织 结构 上 的 好 处 : 
你 的 程序 设计 可 以 极 大 地 简化 。 某 些 类 型 的 问题 ， 例 如 仿真 ， 没 有 并 发 的 支持 是 很 难 解决 的 。 

大 多 数 人 都 看 到 过 至 少 一 种 形式 的 仿真 ， 例 如 计算 机 游戏 或 电影 中 计算 机 生成 的 动画 。 仿 
真 通常 涉及 许多 交互 式 元 素 ， 每 一 个 都 有 “其 自己 的 想法 "。 尽 管 你 可 能 注意 到 了 这 一 点 ， 但 是 
在 单 处 理 器 机 器 上 ， 每 个 仿真 元 素 都 是 由 这 个 处 理 器 驱动 执行 的 ， 从 编程 的 角度 看 ， 模 拟 每 个 
仿真 元 素 都 有 其 自己 的 处 理 器 并 且 都 是 独立 的 任务 ， 这 种 方式 要 容易 得 多 。 

完整 的 仿真 可 能 涉及 非常 大 量 的 任务 ， 这 与 仿真 中 的 每 个 元 素 都 可 以 独立 动作 这 一 事实 相 
对 应 一 一 这 其 中 包含 门 和 岩石 ， 而 不 仅仅 只 是 精灵 和 巫师 。 多 线程 系统 对 可 用 的 线程 数量 的 限 
制 通常 都 会 是 一 个 相对 较 小 的 数字 ， 有 时 就 是 数 十 或 数 百 这 样 的 数量 级 。 这 个 数字 在 程序 控制 
范围 之 外 可 能 会 发 生变 化 一 一 它 可 能 依赖 于 平台 ， 或 者 在 Java 中 ， 依 赖 于 Java 的 版 本 。 在 Java 中 ， 
通常 要 假定 你 不 会 获得 足够 的 线程 ， 从 而 使 得 可 以 为 大 型 仿真 中 的 每 个 元 素 都 提供 一 个 线程 。 

解决 这 个 问题 的 典型 方式 是 使 用 协作 多 线程 。Java 的 线程 机 制 是 抢占 式 的 ， 这 表示 调度 机 
制 会 周期 性 地 中 断 线程 ， 将 上 下 文 切换 到 另 一 个 线程 ， 从 而 为 每 个 线程 都 提供 时 间 片 ， 使 得 每 
个 线程 都 会 分 配 到 数量 合理 的 时 间 去 驱动 它 的 任务 。 在 协作 式 系统 中 ， 每 个 任务 都 会 自动 地 放 
弃 控 制 ， 这 要 求 程序 员 要 有 意识 地 在 每 个 任务 中 插入 某 种 类 型 的 让 步 语句 。 协 作 式 系统 的 优势 
是 双重 的 ， 上 下 文 切换 的 开销 通常 比 抢占 式 系统 要 低廉 许多 ， 并 且 对 可 以 同时 执行 的 线程 数量 
在 理论 上 没有 任何 限制 。 当 你 处 理 大 量 的 仿真 元 素 时 ， 这 可 以 一 种 理想 的 解决 方案 。 但 是 注意 ， 
某 些 协作 式 系统 并 未 设计 为 可 以 在 多 个 处 理 器 之 间 分 布 任务 ， 这 可 能 会 非常 受 限 。 

在 另 一 个 极端 ， 当 你 用 流行 的 消息 系统 工作 时 ， 由 于 消息 系统 涉及 分 布 在 整个 网 络 中 的 多 
台独 立 的 计算 机 ， 因 此 并 发 就 会 成 为 一 种 非常 有 用 的 模型 ， 因 为 它 是 实际 发 生 的 模型 。 在 这 种 
情形 中 ， 所 有 的 进程 都 彼此 完全 独立 地 运行 ， 甚 至 没有 任何 可 能 去 共享 资源 。 但 是 ， 你 仍旧 必 
须 在 进程 间 同步 信息 ， 使 得 整个 消息 系统 不 会 丢失 信息 或 在 错误 的 时 刻 混 进 信息 。 即 使 你 没有 
打算 在 眼前 大 量 使 用 并 发 ， 理 解 并 发 也 会 很 有 用 ， 因 为 你 可 以 掌握 基于 消息 机 制 的 架构 ， 这 些 
架构 在 创建 分 布 式 系统 时 是 更 主要 的 方式 。 

并 发 需要 付出 代价 ， 包 含 复杂 性 代价 ， 但 是 这 些 代价 与 在 程序 设计 、 资 源 负载 均衡 以 及 用 
户 方便 使 用 方面 的 改进 相 比 ， 就 显得 微不足道 了 。 通 常 ， 线 程 使 你 能 够 创建 更 加 松散 耦合 的 设 
计 ， 和 否则 ， 你 的 代码 中 各 个 部 分 都 必须 显 式 地 关注 那些 通常 可 以 由 线程 来 处 理 的 任务 。 


21.2 基本 的 线程 机 制 


并 发 编程 使 我 们 可 以 将 程序 划分 为 多 个 分 离 的 、 独 立 运行 的 任务 。 通 过 使 用 多 线程 机 制 ， 
这 些 独立 任务 (也 被 称 为 子 任务 ) 中 的 每 一 个 都 将 由 执行 线程 来 驱动 。 一 个 线程 就 是 在 进程 中 
的 一 个 单一 的 顺序 控制 流 ， 因 此 ， 单 个 进程 可 以 拥有 多 个 并 发 执行 的 任务 ， 但 是 你 的 程序 使 得 
每 个 任务 都 好 像 有 其 自己 的 CPU 一 样 。 其 底层 机 制 是 切 分 CPU 时 间 ， 但 通常 你 不 需要 考虑 它 。 

线程 模型 为 编程 带 来 了 便利 ， 它 简化 了 在 单一 程序 中 同时 交织 在 一 起 的 多 个 操作 的 处 理 。 
在 使 用 线程 时 , CPU 将 轮流 给 每 个 任务 分 配 其 占用 时 间 ©, 每 个 任务 都 觉得 自己 在 一 直 占用 CPU， 





但 事实 上 CPU 时 间 是 划分 成 片段 分 配给 了 所 有 的 任务 (例外 情况 是 程序 确实 运行 在 多 个 CPU 之 


O 当 系统 使 用 时 间 切 片 机 制 时 ， 人 情况 确实 如 此 【例如 Windows)。Solaris 使 用 了 FIFO 并 发 模型 ， 除 非 有 高 优先 级 
的 线程 被 唤醒 ， 否 则 当前 线程 将 一 直 运 行 ， 直 至 它 被 阻塞 或 终止 。 这 意味 着 具有 相同 优先 级 的 其 他 线程 在 当前 
线程 放弃 处 理 器 之 前 ， 将 不 会 运行 。 
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上 )。 线 程 的 一 大 好 处 是 可 以 使 你 从 这 个 层次 抽身 出 来 ， 即 代码 不 必 知 道 它 是 运行 在 具有 一 个 还 
是 多 个 CPU 的 机 器 上 。 所 以 ， 使 用 线程 机 制 是 一 种 建立 透明 的 、 可 扩展 的 程序 的 方法 ， 如 果 程 
序 运 行 得 太 慢 ， 为 机 器 增添 一 个 CPU 就 能 很 容易 地 加 快 程序 的 运行 速度 。 多 任务 和 多 线程 往往 
是 使 用 多 处 理 器 系统 的 最 合理 方式 。 
21.2.1 定义 任务 

线程 可 以 驱动 任务 ， 因 此 你 需要 一 种 描述 任务 的 方式 ， 这 可 以 由 Runnable 接 口 来 提供 。 要 
想 定 义 任务 ， 只 需 实现 Runnable 接 口 并 编写 run0 方 法 ， 使 得 该 任务 可 以 执行 你 的 命令 。 例 如 ， 
下 面 的 LiftOff 任 务 将 显示 发 射 之 前 的 倒计时 : 


41: concurrency/Liftoff. java 
// Demonstration of the Runnable interface. 


public class Liftoff implements Runnable { 
protected int countDown = 10; // Default 
private static int taskCount = 0; 
private final int id = taskCount++; 
public LiftOff() (} 
public LiftOff(int countdown) { 
this.countDown = countDown; 


} 
public String status() { 
return "#" + id + "(" + 
(countDown > 6 ? countDown : “Liftoff!") +"), "; 


} 
public void run() { 
while(countDown-- > @) { 
System. out. print (status()); 
Thread. yield(); 


} 

} Mi~ 

标识 符 id 可 以 用 来 区 分 任务 的 多 个 实例 ， 它 是 final 的 ， 因 为 它 一 旦 被 初始 化 之 后 就 不 希望 
被 修改 。 

任务 的 run() 方 法 通常 总 会 有 某 种 形式 的 循环 ， 使 得 任务 一 直 运行 下 去 直到 不 再 需要 ， 所 以 
要 设 定 跳出 循环 的 条 件 (有 一 种 选择 是 直接 从 run0 返 回 )。 通 常 ，run0 被 写成 无 限 循环 的 形式 ， 
这 就 意味 着 ， 除 非 有 某 个 条 件 使 得 run0 终 止 ， 否 则 它 将 永远 运行 下 去 〈 在 本 章 后 面 将 会 看 到 如 
何 安全 地 终止 线程 ) 。 

在 run0 中 对 静态 方法 Thread.yield0 的 调用 是 对 线程 调度 器 (Java 线程 机 制 的 一 部 分 ， 可 以 
将 CPU 从 一 个 线程 转移 给 另 一 个 线程 ) 的 一 种 建议 ， 它 在 声明 :“ 我 已 经 执行 完 生 命 周期 中 最 
重要 的 部 分 了 ， 此 刻 正 是 切换 给 其 他 任务 执行 一 段 时 间 的 大 好 时 机 。” 这 完全 是 选择 性 的 ， 但 
是 这 里 使 用 它 是 因为 它 会 在 这 些 示例 中 产生 更 加 有 趣 的 输出 : 你 更 有 可 能 会 看 到 任务 换 进 换 出 
的 证 据 。 

在 下 面 的 实例 中 ， 这 个 任务 的 ran0 不 是 由 单独 的 线程 驱动 的 ， 它 是 在 main0 中 直接 调用 的 
(实际 上 ， 这 里 仍旧 使 用 了 线程 ， 即 总 是 分 配给 main0 的 那个 线程 ): 

/1/: concurrency/MainThread. java 


public class MainThread { 
public static void main(String[] args) { 
Liftoff Launch = new LiftOff(); 
launch.run(); 
} 
} /* Output: 
#0(9), #0(8), #0(7), #0(6), #0(5), #0(4), #0(3), #6(2), 
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#0(1), #O(Liftoff!), 
“i~ 


当 从 Runnable 导 出 一 个 类 时 ， 它 必须 具有 run0 方 法 ， 但 是 这 个 方法 并 无 特殊 之 处 一 一 它 不 
会 产生 任何 内 在 的 线程 能 力 。 要 实现 线程 行为 ， 你 必须 显 式 地 将 一 个 任务 附着 到 线程 上 。 


21.2.2 Thread 类 

将 Runnable 对 象 转变 为 工作 任务 的 传统 方式 是 把 它 提交 给 一 个 Thread 构 造 器 ， 下 面 的 示例 
展示 了 如 何 使 用 Thread 来 驱动 LiftOff 对 象 : 

//: concurrency/BasicThreads. java 

// The most basic use of the Thread class. 


public class BasicThreads { 
public static Void main(String{] args) { 
Thread t = new Thread(new Liftoft()); 
t,start(); 
System.out.printin("Waiting for Liftoff"); 


$ 
} /* Output: (96% match) 
Waiting for Liftoff 
#0(9), #0(8), #0(7), #0(6), #0(5), #0(4), #0(3), #0(2), 
#O(1), #O(Liftoff!), 
Whim 


Thread 构 造 器 只 需要 一 个 Runnable 对 象 。 调 用 Thread 对 象 的 start( 方 法 为 该 线程 执行 必需 
的 初始 化 操作 ， 然 后 调用 Runnable 的 run0 方 法 ， 以 便 在 这 个 新 线程 中 启动 该 任务 。 尽 管 start0 
看 起 来 是 产生 了 一 个 对 长 期 运行 方法 的 调用 ， 但 是 从 输出 中 可 以 看 到 ，start0 迅 速 地 返回 了 ， 因 
为 Waiting for LiftoOff 消 息 在 倒计时 完成 之 前 就 出 现 了 。 实 际 上 ， 你 产生 的 是 对 LiftOff.run0 的 方 
法 调用 ， 并 且 这 个 方法 还 没有 完成 ， 但 是 因为 LiftOff.run0 是 由 不 同 的 线程 执行 的 你 仍旧 
可 以 执行 main0 线 程 中 的 其 他 操作 〈 这 种 能 力 并 不 局 限于 main0 线 程 ， 任 何 线程 都 可 以 启动 另 一 
个 线程 ) 。 ， 程 序 会 同时 运行 两 个 方法 ，main0 和 LiftOff.run0 是 程序 中 与 其 他 线程 “同时 
执行 的 代码 。 
你 可 以 很 容易 地 添加 更 多 的 线程 去 驱动 更 多 的 任务 。 下 面 ， 你 可 以 看 到 所 有 任务 彼此 之 间 
是 如 何 互相 呼应 的 。: 


//: concurrency/MoreBasicThreads. java 
/1 Adding more threads. 








public class MoreBasicThreads { 
public static void main(String{] args) { 
for(int 1 = 0; i < 5; i++) 
new Thread(new Liftoff()).start(); 
System.out.printin(*Waiting for Liftoff"); 


} /* Output: (Sample) 
Waiting for Liftoff 

#0(9), #1(9), #2(9), #3(9), #4(9), #0(8), #1(8), #2(8), 
#3(8), #4(8), #0(7), #1(7). #2(7), #3(7), #4(7), #0(6), 
#1(6), #2(6), #3(6), #4(6), #0(5), #1(5), #2(5), #3(5), 
#4(5), #O(4), #1(4), #2(4), #3(4), #4(4), #0(3), #1(3), 
#2(3), #3(3), #4(3), #0(2), #1(2), #2(2), #3(2), #4(2), 
#0(1), #1(1), #2(1), #3(1), #4(1), #O(Liftoff!), 
#L(Liftoff!), #2(Liftoff!), #3(Liftoff!), #4(Liftoff!), 
Whim 


输出 说 明 不 同 任务 的 执行 在 线程 被 换 进 换 出 时 混在 了 一 起 。 这 种 交换 是 由 线程 调度 器 自动 控 


O 在 本 例 中 ,单一 线程 (main0) 在 创建 所 有 的 LiftOff 线 程 。 但 是 ， 如 果 多 个 线程 在 创建 LiftOff 线 程 ， 那 么 就 有 
可 能 会 有 多 个 LiftOff 拥 有 相同 的 这 。 在 本 章 稍 后 你 会 了 解 到 这 是 为 什么 。 
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制 的 。 如 果 在 你 的 机 器 上 有 多 个 处 理 器 ， 线 程 调度 器 将 会 在 这 些 处 理 器 之 间 默 默 地 分 发 线程 。 

这 个 程序 一 次 运行 的 结果 可 能 与 另 一 次 运行 的 结果 不 同 ， 因 为 线程 调度 机 制 是 非 确定 性 的 。 
事实 上 ， 你 可 以 看 到 ， 在 某 个 版 本 的 JDK 与 下 个 版 本 之 间 ， 这 个 简单 程序 的 输出 会 产生 巨大 的 
差异 。 例 如 ， 较 旱 的 JDK 不 会 频繁 对 时 间 切片 ， 因 此 线程 1 可 能 会 首先 循环 到 尽头 ， 然 后 线程 2 
会 经 历 其 所 有 循环 ， 等 等 。 这 实际 上 与 调用 一 个 例 程 去 同时 执行 所 有 的 循环 一 样 ， 只 是 启动 所 
有 线程 的 代价 要 更 加 高 昂 。 较 晚 的 JDK 看 起 来 会 产生 更 好 的 时 间 切 片 行为 ， 因 此 每 个 线程 看 起 
来 都 会 获得 更 加 正规 的 服务 。 通 常 ，Sun 并 为 提 及 这 些 种 类 的 JDK 的 行为 变化 ， 因 此 你 不 能 依赖 
于 任何 线程 行为 的 一 致 性 。 最 好 的 方式 是 在 编写 使 用 线程 的 代码 时 ， 尽 可 能 地 保守 。 

当 mainO 创 建 Thread 对 象 时 ， 它 并 没有 捕获 任何 对 这 些 对 象 的 引用 。 在 使 用 普通 对 象 时 ， 
这 对 于 垃圾 回收 来 说 是 一 场 公平 的 游戏 ， 但 是 在 使 用 Thread 时 ， 情 况 就 不 同 了 。 每 个 Thread 都 
“注册 ”了 它 自己 ， 因 此 确实 有 一 个 对 它 的 引用 ， 而 且 在 它 的 任务 退出 其 run0 并 死亡 之 前 ,垃圾 
回收 器 无 法 清除 它 。 你 可 以 从 输出 中 看 到 ， 这 些 任务 确实 运行 到 了 结束 ， 因 此 ， 一 个 线程 会 创 
建 一 个 单独 的 执行 线程 ， 在 对 start0 的 调用 完成 之 后 ， 它 仍旧 会 继续 存在 。 

练习 1: (2) 实现 一 个 Runnable。 在 run0 内 部 打印 一 个 消息 ， 然 后 调用 yield0。 重 复 这 个 操 
作 三 次 ， 然 后 从 run0 中 返回 。 在 构造 器 中 放置 一 条 启动 消息 ， 并 且 放 置 一 条 在 任务 终止 时 的 关 
闭 消息 。 使 用 线程 创建 大 量 的 这 种 任务 并 驱动 它们 。 

练习 2: (2) 遵循 gererie/Fibonaccijava 的 形式 ， 创 建 一 个 任务 , CTL Az Ham Mee eae 
字 组 成 的 序列 ， 其 中 m 是 通过 任务 的 构造 器 而 提供 的 。 使 用 线程 创建 大 量 的 这 种 任务 并 驱动 它们 。 
21.2.3 使 用 Executor 

Java SE5 的 java.util.concurrent 包 中 的 执行 器 (Executor) 将 为 你 管理 Thread 对 象 ， 从 而 简 
化 了 并 发 编程 。Executor 在 客户 端 和 任务 执行 之 间 提 供 了 一 个 间接 层 ， 与 客户 端 直接 执行 任务 
不 同 ， 这 个 中 介 对 象 将 执行 任务 。Executor 人 允许 你 管理 异步 任务 的 执行 ， 而 无 须 显 式 地 管理 线 
程 的 生命 周期 。Executor 在 Java SE5/6 中 是 启动 任务 的 优选 方法 。 

我 们 可 以 使 用 Executor 来 代替 在 MoreBasicThreads.java 中 显示 地 创建 Thread 对 象 。LiftOff 
对 象 知道 如 何 运 行 具体 的 任务 ， 与 命令 设计 模式 一 样 ， 它 暴露 了 要 执行 的 单一 方法 。 
ExecutorService (具有 服务 生命 周期 的 Executor， 例 如 关闭 ) 知道 如 何 构建 恰当 的 上 下 文 来 执 
行 Runnable 对 象 。 在 下 面 的 示例 中 ，CachedThreadPool 将 为 每 个 任务 都 创建 一 个 线程 。 注 意 ， 
ExecutorService 对 象 是 使 用 静态 的 Executor 方 法 创建 的 ， 这 个 方法 可 以 确定 其 Executor 类 型 ， 


//: concurrency/CachedThreadPool. java 
import java.util.concurrent.*; 


public class CachedThreadPool { 
public static void main(String[] args) { 
ExecutorService exec = Executors.newCachedThreadPool () ; 
for(int 1 = 0; 1 < 5; i++) 
exec.execute(new LiftOff()); 
exec. shutdown () ; 
} 
} /* Output: (Sample) 
#0(9), #0(8), #1(9), #2(9), #3(9), #4(9), #8(7), #1(8), 
#2(8), #3(8), #4(8), #0(6), #1(7), #2(7), #3(7), #4(7), ` 
#0(5). #1(6), #2(6), #3(6), #4(6), #0(4), #1(5), #2(5), 
#3(5), #4(5), #0(3), #1(4), #2(4), #3(4), #4(4), #002), 
#1(3), #2(3), #3(3), #4(3), #0(1), #1(2), #2(2), #302), 
#4(2), #0(L1ftoff!), #1(1), #2(1), #3(1), #4(1), 


3 


O 对 于 某 些 最 早 版 本 的 Java 来 说 ， 情 况 并 非 如 此 。 





# 发 657 





#1(Liftoff!), #2(Liftoff!), #3(Liftoff!), #4(Liftoff!), 
hm 


非常 常见 的 情况 是 ， 单 个 的 Executor 被 用 来 创建 和 管理 系统 中 所 有 的 任务 。 

对 shutdown0 方 法 的 调用 可 以 防止 新 任务 被 提交 给 这 个 Executor， 当 前 线程 (在 本 例 中 ， 即 
驱动 main() 的 线程 ) 将 继续 运行 在 shutdown() 被 调用 之 前 提交 的 所 有 任务 。 这 个 程序 将 在 
Executor 中 的 所 有 任务 完成 之 后 尽快 退出 。 

你 可 以 很 容易 地 将 前 面 示例 中 的 CachedThreadPool 窒 换 为 不 同类 型 的 Executor。 
FixedThreadPool 使 用 了 有 限 的 线程 集 来 执行 所 提交 的 任务 : 


11: concurrency/FixedThreadPool. java 
import java.util.concurrent.*; 


public class FixedThreadPool { 
public static void main(String[] args) { 
// Constructor argument is number of threads: 
ExecutorService exec = Executors.newFixedThreadPool (5) ; 
for(int i = 6; 1 < 5; i++) 
exec.execute(new Liftoff()): 
exec. shutdown () ; 


} 
} /* Output; (Sample) 
#0(9), #0(8), #1(9), #2(9), #3(9), #4(9), #0(7), #1(8), 
#2(8), #3(8), #4(8), #0(6), #1(7), #2(7), #3(7), #4(7), 
#0(5), #1(6), #2(6), #3(6), #4(6), #0(4), #1(5), #2(5), 
#3(5), #4(5), #O(3), #1(4), #2(4), #3(4), #4(4), #002), 
#1(3), #2(3), #3(3), #4(3), #O(1), #1(2), #2(2), #3(2), 
#4(2), #O(LIftoff!), #1(1), #2(1), #3(1), #4(1), 
#1(Liftoff!), #2(Liftoff!), #3(Liftoff!), #4(Liftoff!), 
Whim 


有 了 FixedThreadPool， 你 就 可 以 一 次 性 预先 执行 代价 高 昂 的 线程 分 配 ， 因 而 也 就 可 以 限 
制 线程 的 数量 了 。 这 可 以 节省 时 间 ， 因 为 你 不 用 为 每 个 任务 都 固定 地 付出 创建 线程 的 开销 。 在 
事件 驱动 的 系统 中 ， 需 要 线程 的 事件 处 理 器 ， 通 过 直接 从 池 中 获取 线程 ， 也 可 以 如 你 所 愿 地 尽 
快 得 到 服务 。 你 不 会 滥用 可 获得 的 资源 ， 因 为 FixedThreadPool 使 用 的 Thread 对 象 的 数量 是 有 
界 的 。 

注意 ， 在 任何 线程 池 中 ， 现 有 线程 在 可 能 的 情况 下 ， 都 会 被 自动 复 用 。 

尽管 本 书 将 使 用 CachedThreadPool， 但 是 也 应 该 考虑 在 产生 线程 的 代码 中 使 用 
FixedThreadPool。CachedThreadPool 在 程序 执行 过 程 中 通常 会 创建 与 所 需 数量 相同 的 线程 ， 然 
后 在 它 回收 旧 线程 时 停止 创建 新 线程 ， 因 此 它 是 合理 的 Executor 的 首选 。 只 有 当 这 种 方式 会 引 
发 问题 时 ， 你 才 需 要 切换 到 FixedThreadPool。 

SingleThreadExecutor 就 像 是 线程 数量 为 1 的 FixedThreadPools 。 这 对 于 你 希望 在 另 一 个 线 
程 中 连续 运行 的 任何 事物 (长 期 存活 的 任务 ) 来 说 ， 都 是 很 有 用 的 ， 例 如 监听 进入 的 套 接 字 连 
接 的 任务 。 它 对 于 希望 在 线程 中 运行 的 短 任务 也 同样 很 方便 ， 例 如 ， 更 新 本 地 或 远程 日 志 的 小 
任务 ， 或 者 是 事件 分 发 线程 。 

如 果 向 SingleThreadExecutor 提 交 了 多 个 任务 ， 那 么 这 些 任务 将 排队 ， 每 个 任务 都 会 在 下 一 
个 任务 开始 之 前 运行 结束 ， 所 有 的 任务 将 使 用 相同 的 线程 。 在 下 面 的 示例 中 ， 你 可 以 看 到 每 个 
任务 都 是 按照 它们 被 提交 的 顺序 ， 并 且 是 在 下 一 个 任务 开始 之 前 完成 的 。 因 此 ，SingleThread- 
了 PExecutor 会 序列 化 所 有 提交 给 它 的 任务 ， 并 会 维护 它 自己 〈 隐 藏 ) 的 悬挂 任务 队列 。 


O 它 还 提供 了 一 种 重要 的 并 发 保证 ， 其 他 线程 不 会 ( 即 没有 两 个 线程 会 ) 被 并 发 调用 。 这 会 改变 任务 的 加 锁 需 求 
(你 将 在 本 章 稍 后 学 习 锁 机 制 )。 
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//: concurrency/SingleThreadExecutor. java 
import java.util.concurrent.*; 


public class SingleThreadExecutor { 
public static void main(Stringl] args) { 
ExecutorService exec = 
Executors.newSingleThreadExecutor (); 
for(int i = @; 1 < 5; i++) 
exec. execute(new LiftOff()); 
exec. shutdown() ; 


} 
} /* Output: 
#0(9), #0(8), #0(7), #0(6), #0(5), #0(4), #0(3), #0(2), 
#O(1), #O(LIftoff!), #1(9), #1(8), #1(7), #1(6), #1(5), 
#1(4), #1(3), #1(2), #1(1), #1(Liftoff!), #2(9), #2(8), 
#2(7), #2(6), #2(5), #2(4), #2(3), #2(2), #2(1), 
#2(Liftoff!), #3(9), #3(8), #3(7), #3(6), #3(5), #3(4), 
#3(3), #3(2), #3(1), #3(Liftoff!), #4(9), #4(8), #4(7), 
#4(6), #4(5), #4(4), #4(3), #4(2), #4(1), #4(Liftoff!), 
Whim 


作为 另 一 个 示例 ， 假 设 你 有 大 量 的 线程 ， 那 它们 运行 的 任务 将 使 用 文件 系统 。 你 可 以 用 


[1123] SingleThreadExecutor 来 运行 这 些 线程 , 以 确保 任意 时 刻 在 任何 线程 中 都 只 有 唯一 的 任务 在 运行 。 


在 这 种 方式 中 ， 你 不 需要 在 共享 资源 上 处 理 同步 〈 同 时 不 会 过 度 使 用 文件 系统 )。 有 时 更 好 的 解 
决 方案 是 在 资源 上 同步 (你 将 在 本 章 稍 后 学 习 ) ， 但 是 SingleThreadExecutor 可 以 让 你 省 去 只 是 
为 了 维持 某 些 事物 的 原型 而 进行 的 各 种 协调 努力 。 通 过 序列 化 任务 ， 你 可 以 消除 对 序列 化 对 象 
的 需求 。 

练习 3: (1) 使 用 本 节 展 示 的 各 种 不 同类 型 的 执行 器 重复 练习 1。 

练习 4: (1) 使 用 本 节 展 示 的 各 种 不 同类 型 的 执行 器 重复 练习 2。 
21.2.4 从 任务 中 产生 返回 值 

Runnable 是 执行 工作 的 独立 任务 ， 但 是 它 不 返回 任何 值 。 如 果 你 希望 任务 在 完成 时 能 够 返 
回 一 个 值 ， 那 么 可 以 实现 Callable 接 口 而 不 是 Runnable 接 口 。 在 Java SE5 中 引入 的 Callabel 是 一 
种 具有 类 型 参数 的 泛 型 ， 它 的 类 型 参数 表示 的 是 从 方法 call0 (而 不 是 run0) 中 返回 的 值 ， 并 且 
必须 使 用 ExecutorService.submit0 方 法 调用 它 ， 下 面 是 一 个 简单 示例 : 


//: concurrency/CallableDemo. java 
import java.util.concurrent.*; 
import java.util.*; 


class TaskWithResult implements Callable<String> { 
private int id; 
public TaskWithResult(int id) { 
this.id = id; 


} 
public String call() { 
return "result of TaskWithResult " + id; 
} 
} 


public class CallableDemo { 
public static void main(String[] args) { 
ExecutorService exec = Executors.newCachedThreadPool () ; 
ArrayList<Future<String>> results = 
new ArrayList<Future<String>>(); 
for(int i = 0; i < 10; i++) 
results. add(exec.submit(new TaskWithResult(i))): 
for (Future<String> fs : results) 
try { 
/1/ get() blocks until completion: 
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System.out.printtn(fs.get()); 
catch(InterruptedException e) { 
System.out.printin(e); 

return; 

} catch(ExecutionException e) { 
System. out.printin(e); 

finally { 

exec. shutdown() ; 


} 


} 

} /* Output: 

result of TaskWithResult 
result of TaskWithResult 
result of TaskWithResult 
result of TaskWithResult 
result of TaskWithResult 
result of TaskWithResult 
result of TaskWithResult 
result of TaskWithResult 
result of TaskWithResult 
result of TaskWithResult 
Wham 


submit0 方 法 会 产生 Future 对 象 ， 它 用 Callable 返 回 结果 的 特定 类 型 进行 了 参数 化 。 你 可 以 
用 isDone0 方 法 来 查询 Future 是 否 已 经 完成 。 当 任务 完成 时 ， 它 具有 一 个 结果 ， 你 可 以 调用 get0 
方法 来 获取 该 结果 。 你 也 可 以 不 用 isDoneO 进 行 检 查 就 直接 调用 get0， 在 这 种 情况 下 ，get0 将 阻 
塞 ， 直 至 结果 准备 就 绪 。 你 还 可 以 在 试图 调用 get0 来 获取 结果 之 前 ， 先 调用 具有 超时 的 get()， 
或 者 调用 isDone0 来 查看 任务 是 否 完成 。 

练习 5: (2) 修改 练习 2， 使 得 计算 所 有 斐 波 纳 契 数字 的 数值 总 和 的 任务 成 为 Callable。 创 建 
多 个 任务 并 显示 结果 。 
21.2.5 休眠 

影响 任务 行为 的 一 种 简单 方法 是 调用 sleep0， 这 将 使 任务 中 止 执行 给 定 的 时 间 。 在 LiftOff 
类 中 ， 要 是 把 对 yield0 的 调用 换 成 是 调用 sleepO， 将 得 到 如 下 结果 : 


//: concurrency/SleepingTask. java 
// Calling sleep() to pause for a while. 
import java.util.concurrent.*; 


COVauaunro 


public class SleepingTask extends Liftoff { 
public void run() { 
try { 
while(countDown-- > 8) { 

System.out.print(status()); 
// Old-style: 
// Thread. steep (109) ; 
// Java SES/6-style: 
TimeUnit MILLISECONDS. sleep (188) ; 


} 
} catch(InterruptedException e) { 
System.err.printtn("Interrupted"); 
} 
} 
public static void main(String{] args) { 
ExecutorService exec = Executors.newCachedThreadPool (); 
for(int i = 6: 1 < 5; i++) 
exec.execute(new SleepingTask()); 
exec. shutdown(); 


} 
} /* Output: 
#0(9), #1(9), #2(9), #3(9), #4(9), #0(8), #1(8), #2(8). 
#3(8), #4(8), #0(7), #1(7), #2(7), #3(7), #4(7), #0(6), 
#1(6), #2(6), #3(6), #4(6), #0(5), #1(5), #2(5). #3(5), 
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#4(5), #0(4), #1(4), #2(4), #3(4), #4(4), #93), #13), 
#2(3), #3(3), #4(3), #0(2), #1(2), #2(2), #3(2), #4(2), 
#O(1), #1(1), #2(1), #3(1), #4(1), #O(Liftoff!), 
#L(Liftoff!), #2(Liftoff!), #3(Liftoff!), #4(Liftoff!), 
Wh 


对 sleep0 的 调用 可 以 抛 出 InterruptedException 异 常 ， 并 且 你 可 以 看 到 ， 它 在 run0 中 被 捕获 。 


因为 异常 不 能 跨 线程 传播 回 main0， 所 以 你 必须 在 本 地 处 理 所 有 在 任务 内 部 产生 的 异常 。 








1127] 








Java SE5 引 入 了 更 加 显 式 的 sleep0 版 本 ， 作 为 TimeUnit 类 的 一 部 分 ， 就 像 上 面 示例 所 示 的 那 
样 。 这 个 方法 允许 你 指定 sleep0 延 迟 的 时 间 单 元 ， 因 此 可 以 提供 更 好 的 可 阅读 性 。TimeUnit 还 
可 以 被 用 来 执行 转换 ， 就 像 稍 后 你 会 在 本 书 中 看 到 的 那样 。 

你 可 能 会 注意 到 ， 这 些 任务 是 按照 “完美 的 分 布 ”顺序 运行 的 ， 即 从 0 到 4， 然 后 再 回 过 头 
从 0 开始 ， 当 然 这 取决 于 你 的 平台 。 这 是 有 意义 的 ， 因 为 在 每 个 打印 语句 之 后 ， 每 个 任务 都 将 要 
睡眠 〈 即 阻塞 ) ， 这 使 得 线程 调度 器 可 以 切换 到 另 一 个 线程 ， 进 而 驱动 另 一 个 任务 。 但 是 ， 顺 序 
行为 依赖 于 底层 的 线程 机 制 ， 这 种 机 制 在 不 同 的 操作 系统 之 间 是 有 差异 的 ， 因 此 ， 你 不 能 依赖 
于 它 。 如 果 你 必须 控制 任务 执行 的 顺序 ， 那 么 最 好 的 押宝 就 是 使 用 同步 控制 ( 稍 候 描述 ) ， 或 者 
在 某 些 情况 下 ， 压 根 不 使 用 线程 ， 但 是 要 编写 自己 的 协作 例 程 ， 这 些 例 程 将 会 按照 指定 的 顺序 
在 互相 之 间 传递 控制 权 。 

练习 6: (2) 创建 一 个 任务 ， 它 将 睡眠 1 至 10 秒 之 间 的 随机 数量 的 时 间 ， 然 后 显示 它 的 睡眠 时 
间 并 退出 。 创 建 并 运行 一 定数 量 的 这 种 任务 。 

21.2.6 优先 级 

线程 的 优先 级 将 该 线程 的 重要 性 传递 给 了 调度 器 。 尽 管 CPU 处 理 现 有 线程 集 的 顺序 是 不 确 
定 的 ， 但 是 调度 器 将 倾向 于 让 优先 权 最 高 的 线程 先 执行 。 然 而 ， 这 并 不 是 意味 着 优先 权 较 低 的 
线程 将 得 不 到 执行 〈 也 就 是 说 ， 优 先 权 不 会 导致 死 锁 ) 。 优 先 级 较 低 的 线程 仅仅 是 执行 的 频率 
较 低 。 

在 绝 大 多 数 时 间 里 ， 所 有 线程 都 应 该 以 默认 的 优先 级 运行 。 试 图 操纵 线程 优先 级 通常 是 一 
种 错误 。 

下 面 是 一 个 演示 优先 级 等 级 的 示例 ， 你 可 以 用 getPriority0 来 读 取现 有 线程 的 优先 级 ， 并 且 
在 任何 时 刻 都 可 以 通过 setPriority0 来 修改 它 。 


//: concurrency/SimplePriorities. java 
// Shows the use of thread priorities. 
import java.util.concurrent.*; 


public class SimplePriorities implements Runnable { 
private int countdown = 5; 
private volatile double d; // No optimization 
private int priority: 
public SimplePriorities(int priority) ( 
this.priority = priority; 


} 
public String tostring() { 
return Thread.currentThread() + ": ”+ CountDown: 


} 
Public void run() { 
Thread. currentThread().setPriority (priority) ; 
while(true) { 
// An expensive, interruptable operation: 
for(int i = 1; 1 < 100080; i++) { 
d += (Math.PI + Math.£) / (double)i; 
if (i % 1000 == 0) 





Thread.yield(); 
} 
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System. out.println(this); 
if(--countDown == 6) return; 


} 


} 
public static void main(String(} args) { 
ExecutorService exec = Executors.newCachedThreadPool () ; 
for(int i = @: i < 5; i++) 
exec. execute( 
new SimplePriorities(Thread.MIN_PRIORITY)); 
exec. execute ( 
new SimplePriorities (Thread. MAX_PRIORITY) ) ; 
exec. shutdown () ; 


} 

} /* Output: (76% match) 

Thread{pool-1-thread-6,10,main] : 
Thread[pool-1-thread-6,18,main] : 
Thread{pool- 
Thread{poot-1- 
Thread {pool - 
Thread {pool - 
Thread{pool- 
Thread (pool- 
Thread[pool-1-thread-5,1,main]: 5 
Thread[pool-1-thread-4,1,main]: 5 


wwaew 





“~ 

toString0 方 法 被 覆盖 ， 以 便 使 用 Thread.toString0 方 法 来 打印 线程 的 名 称 、 线 程 的 优先 级 以 
及 线程 所 属 的 ”线程 组 "。 你 可 以 通过 构造 器 来 自己 设置 这 个 名 称 ， 这 里 是 自动 生成 的 名 称 ， 如 
pool-1-thread-1，pool-1-thread-2 等 。 和 覆盖 后 的 toString0 方 法 还 打印 了 线程 的 倒 计 数值 。 注 意 ， 
你 可 以 在 一 个 任务 的 内 部 ， 通 过 调用 Thread.currentThread0 来 获得 对 驱动 该 任务 的 Thread 对 象 
的 引用 。 

可 以 看 到 ， 最 后 一 个 线程 的 优先 级 最 高 ， 其 余 所 有 线程 的 优先 级 被 设 为 最 低 。 注 意 ， 优 先 
级 是 在 ran0 的 开头 部 分 设 定 的 ， 在 构造 器 中 设置 它们 不 会 有 任何 好 处 ， 因 为 Executor 在 此 刻 还 
没有 开始 执行 任务 。 

在 run0 里 ， 执 行 了 100 000 次 开销 相当 大 的 浮 点 运算 ， 包 括 double 类 型 的 加 法 与 除法 。 变 量 
4 是 volatile 的 ， 以 努力 确保 不 进行 任何 编译 器 优化 。 如 果 没 有 加 入 这 些 运算 的 话 ， 就 看 不 到 设置 
优先 级 的 效果 ( 试 一 试 ， 把 包含 double 运 算 的 for 循 环 注释 掉 )。 有 了 这 些 运 算 ， 就 能 观察 到 优先 
级 为 MAX_PRIORITY 的 线程 被 线程 调度 器 优先 选择 (至 少 在 我 的 Windows XP 机 器 上 是 这 样 )。 
尽管 向 控制 台 打 印 也 是 开销 较 大 的 操作 ， 但 在 那 种 情况 下 看 不 出 优先 级 的 效果 ， 因 为 向 控制 台 
打印 不 能 被 中 断 (否则 的 话 ， 在 多 线程 情况 下 控制 台 显示 就 乱 套 了 )， 而 数学 运算 是 可 以 中 断 的 。 
这 里 运算 时 间 足 够 的 长 ， 因 此 线程 调度 机 制 才 来 得 及 介入 ， 交 换 任 务 并 关注 优先 级 ， 使 得 最 高 
优先 级 线程 被 优先 选择 。 

尽管 JDK 有 10 个 优先 级 但 它 与 多 数 操作 系统 都 不 能 映射 得 很 好 。 比 如 ，Windows 有 7 个 优 
先 级 且 不 是 固定 的 ， 所 以 这 种 映射 关系 也 是 不 确定 的 。Sun 的 Solaris 有 2 个 优先 级 。 唯 一 可 移植 
的 方法 是 当 调整 优先 级 的 时 候 ， 只 使 用 MAX_PRIORITY、NORM_PRIORITY 和 
MIN_PRIORITY 三 种 级 别 。 
21.2.7 让 步 

如 果 知 道 已 经 完成 了 在 run0 方 法 的 循环 的 一 次 先 代 过 程 中 所 需 的 工作 ， 就 可 以 给 线程 调度 
机 制 一 个 暗示 : 你 的 工作 已 经 做 得 差不多 了 ， 可 以 让 别 的 线程 使 用 CPU 了 。 这 个 暗示 将 通过 调 
用 yield0 方 法 来 作出 (不 过 这 只 是 一 个 上 暗示， 没有 任何 机 制 保证 它 将 会 被 采纳 ) 。 当 调用 yield0 
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时 ， 你 也 是 在 建议 具有 相同 优先 级 的 其 他 线程 可 以 运行 。 

LiftOff.java 使 用 yield0 在 各 种 不 同 的 LiftOff 任 务 之 间 产 生 分 布 良 好 的 处 理 机 制 。 尝 试 着 注 
释 掉 LiftOff.run0 中 的 Thread.yield0， 以 查看 区 别 。 但 是 ， 大 体 上 ， 对 于 任何 重要 的 控制 或 在 调 
整 应 用 时 ， 都 不 能 依赖 于 yield0。 实 际 上 ，yield0 经 常 被 误 用 。 
21.2.8 后 台 线 程 

所 谓 后 台 (daemon) 线程 ， 是 指 在 程序 运行 的 时 候 在 后 台 提供 一 种 通用 服务 的 线程 ， 并 且 
这 种 线程 并 不 属于 程序 中 不 可 或 缺 的 部 分 。 因 此 ， 当 所 有 的 非 后 台 线 程 结束 时 ， 程 序 也 就 终止 
了 ， 同 时 会 杀 死 进程 中 的 所 有 后 台 线程 。 反 过 来 说 ， 只 要 有 任何 非 后台 线 程 还 在 运行 ， 程 序 就 
不 会 终止 。 比 如 ， 执 行 main0 的 就 是 一 个 非 后 台 线 程 。 

1/: concurrency/SimpleDaemons. java 

// Daemon threads don't prevent the program from ending. 


import java.util.concurrent.*; 
import static net.mindview.util.Print.*; 





public class SimpleDaemons implements Runnable { 
public void run() { 
try { 
while(true) { 
TimeUnit .MILLISECONDS .sleep(10@) ; 
print(Thread.currentThread() + " " + this); 


} catch(InterruptedException e) { 
print("sleep() interrupted”); 
} 


public static void main(String[] args) throws Exception { 
for(int i = @; 1 < 10; i++) { 
Thread daemon = new Thread(new SimpleDaemons()); 
daemon. setDaemon(true): // Must call before start() 
daemon. start(); 


) 
print("ALl daemons started"); 
TimeUnit MILLISECONDS. sleep(175) ; 


} 

} /* Output: (Sample) 

All daemons started 

ThreadIThread-6,5,main] SimpleDaemons@530daa 
Thread [Thread SimpleDaemons@a62fc3 
Thread(Thread in] SimpleDaemons@89ae9e 
Thread{Thread-3 in] SimpteDaemons@1278b73 
Thread{Thread-4 in] SimpleDaemons@6@aeb@ 
Thread(Thread-5,5,main) SimpleDaemons@16caf43 
Thread[Thread-6,5.main] SimpleDaemons@66848c 
Thread[Thread-7,5,main] SimpleDaemons@88132 
Thread(Thread-8,5,main] SimpleOaemons@idS6aae 
Thread(Thread-9,5,main] SimpleDaemons@83cc67 









Whi~ 

必须 在 线程 启动 之 前 调用 setDaemon0 方 法 ， 才 能 把 它 设置 为 后 台 线 程 。 

一 旦 mainO 完 成 其 工作 ， 就 没什么 能 阻止 程序 终止 了 ， 因 为 除了 后 台 线程 之 外 ， 已 经 没有 
线程 在 运行 了 。main() 线 程 被 设 定 为 短暂 睡眠 ， 所 以 可 以 观察 到 所 有 后 台 线 程 启动 后 的 结果 。 
不 这 样 的 话 ， 你 就 只 能 看 见 一 些 后 台 线 程 创建 时 得 到 的 结果 〈 试 试 调整 sleep0 休 眠 的 时 间 ， 以 
观察 这 个 行为 ) 。 

SimpleDaemons.java 创 建 了 显 式 的 线程 ， 以 便 可 以 设置 它们 的 后 台 标 志 。 通 过 编写 定制 的 
ThreadFactory 可 以 定制 由 Executor 创 建 的 线程 的 属性 (后台 、 优 先 级、 名 称 ) : 
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/1/: net/mindview/util/DaemonThreadFactory. java 
package net.mindview.util; 
import java.util.concurrent.*; 


public class DaemonThreadFactory implements ThreadFactory { 
public Thread newThread(Runnable r) { 
Thread t = new Thread(r); 
t.setDaemon(true) ; 
return t: 
} 
yh 


这 与 普通 的 ThreadFactory 的 唯一 差异 就 是 它 将 后 台 状 态 全 部 设置 为 了 true。 你 现在 可 以 用 
一 个 新 的 DaemonThreadFactory 作 为 参数 传递 给 Executor.newCachedThreadPool0: 


/1/: concurrency/DaemonFromFactory . java 

// Using a Thread Factory to create daemons. 
import java.util.concurrent.*; 

import net.mindview.util. 
import static net.mindview.util.Print.*; 








public class DaemonFromFactory implements Runnable { 
public void run() { 
try { 
while(true) { 
TimeUnit MILLISECONDS. sleep(160) 
print(Thread.currentThread() + ” * + this); 
} 
} catch(InterruptedException e) { 
print("Interrupted"); 
} 


} 
public static void matn(String[] args) throws Exception { 
ExecutorService exec = Executors.newCachedThreadPool ( 
new DaemonThreadFactory()) ; 
for(int i = @; 1 < 10; i++) 
exec. execute(new DaemonFromFactory()) ; 
print("All daemons started"); 
TimeUnit MILLISECONDS. sleep(50@); // Run for a while 
} 
} /* (Execute to see output) *///:~ 


每 个 静态 的 ExecutorService 创 建 方法 都 被 重 载 为 接受 一 个 ThreadFactory 对 象 ， 而 这 个 对 象 
将 被 用 来 创建 新 的 线程 : 
//: net/mindview/util/DaemonThreadPoolExecutor .java 


package net.mindview.util; 
import java.util.concurrent 





public class DaemonThreadPoolExecutor 
extends ThreadPoolExecutor { 
public DaemonThreadPoolExecutor() { 1132} 














super(®, Integer.MAX_VALUE, 68L, TimeUnit. SECONDS, 
new SynchronousQueue<Runnable>() , 
new DaemonThreadFactory()); 
》 
} M~ 


可 以 通过 调用 isDaemon( 方 法 来 确定 线程 是 否 是 一 个 后 台 线程 。 如 果 是 一 个 后 台 线程 ， 那 
么 它 创建 的 任何 线程 将 被 自动 设置 成 后 台 线 程 ， 如 下 例 所 示 : 


//: concurrency/Daemons. java 

// Daemon threads spawn other daemon threads. 
import java.util.concurrent.*; 

import static net.mindview.util.Print.*; 
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class Daemon implements Runnable { 
private Thread{] t = new Thread{10]; 
public void run() { 
for(int i = @; i < t.length; i++) { 
tli] = new Thread(new DaemonSpawn()); 
tli].startQ: 
printnb("DaemonSpawn " + i +" started, "); 
} 
for(int 1 = @; i < t.length; i++) 
printnb("t(" + i + "].isDaemon() = * + 
tli] .isDaemon() +", *); 
while(true) 
Thread. yield(); 





} 
} 


class DaemonSpawn implements Runnable { 
public void run() { 
while(true) 
Thread.yield(); 
} 


publi¢ class Daemons { 
public static void main(String{] args) throws Exception { 

Thread d = new Thread(new Daemon()); 

d. setDaemon(true) ; 

d.startQ); 

printnb("d.isDaemon() = ”+ d.isDaemon() + *, "); 

// Allow the daemon threads to 

// finish their startup processes: 

TimeUnit. SECONDS. sleep(1); 

} 

} /* Output: (Sample) 
d.isDaemon() = true, DaemonSpawn @ started, DaemonSpawn 1 
started, DaemonSpawn 2 started, DaemonSpawn 3 started, 
DaemonSpawn 4 started. DaemonSpawn 5 started, DaemonSpawn 6 
started, DaemonSpawn 7 started, DaemonSpawn 8 started, 
DaemonSpawn 9 started, t[0].isDaemon() = true, 
t{1].isDaemon() = true, t{2].isDaemon() = true, 
t(3].isDaemon() = true, t[4).isDaemon() = true, 
t{5].isDaemon() = true, t[6].isDaemon() = true, 
t[7] .isDaemon() = true, t{8].isDaemon() = true, 
t[9].isDaemon() = true, 
“Whim 


Daemon 线 程 被 设置 成 了 后 台 模 式 ， 然 后 派生 出 许多 子 线程 ， 这 些 线程 并 没有 被 显 式 地 设置 为 后 
台 模 式 ， 不 过 它们 的 确 是 后 台 线程 。 接 着 ，Daemon 线 程 进入 了 无 限 循 环 ， 并 在 循环 里 调用 
yield0 方 法 把 控制 权 交 给 其 他 进程 。 

你 应 该 意识 到 后 台 进程 在 不 执行 finally 子 名 的 情况 下 就 会 终止 其 run0 方 法 : 


//: concurrency/DaemonsDontRunFinally,. java 

// Daemon threads don't run the finally clause 
import java.util.concurrent.*; 

‘import static net.mindview.util.Print.*: 


class ADaemon implements Runnable { 
public void run() { 
try { 
print("Starting ADaemon") ; 
TimeUnit.. SECONDS. sleep(1 
} catch(InterruptedException e) { 
print("Exiting via InterruptedException); 
} finally { 
print("This should always run?"); 
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} 
} 
} 


public class DaemonsDontRunFinally { 
public static void main(String{] args) throws Exception { 
Thread t = new Thread(new ADaemon()); 
t.setDaemon(true); 
t.start(); 


和 I Output: 

Starting ADaemon 

Wh 

当 你 运行 这 个 程序 时 ， 你 将 看 到 finally 子 名 就 不 会 执行 ， 但 是 如 果 你 注释 掉 对 setDaemon() 
的 调用 ， 就 会 看 到 finally 子 句 将 会 执行 。 

这 种 行为 是 正确 的 ， 即 便 你 基于 前 面 对 finally 给 出 的 承诺 ， 并 不 希望 出 现 这 种 行为 ， 但 情况 
仍 将 如 此 。 当 最 后 一 个 非 后 台 线 程 终止 时 ， 后 台 线 程 会 “突然 ”终止 。 因 此 一 旦 main() 退 出 ， 
JVM 就 会 立即 关闭 所 有 的 后 台 进程 ， 而 不 会 有 任何 你 希望 出 现 的 确认 形式 。 因 为 你 不 能 以 优雅 
的 方式 来 关闭 后 台 线 程 ， 所 以 它们 几乎 不 是 一 种 好 的 思想 。 非 后 台 的 Executor 通 常 是 一 种 更 好 
的 方式 ， 因 为 Executor 控 制 的 所 有 任务 可 以 同时 被 关闭 。 正 如 你 将 要 在 本 章 稍 后 看 到 的 ， 在 这 
种 情况 下 ， 关 闭 将 以 有 序 的 方式 执行 。 

练习 7:， (2) 在 Daemonsjava 中 使 用 不 同 的 休眠 时 间 ， 并 观察 结果 。 ， 

练习 8: (1) 把 SimpleThread.java 中 的 所 有 线程 修改 成 后 台 线程 ， 并 验证 一 旦 main0 退 出 ， 
程序 立刻 终止 。 

练习 9，(3) 修改 SimplePrioritiesjava， 使 得 定制 的 ThreadFactory 可 以 设置 线程 的 优先 级 。 
21.2.9 编码 的 变 体 ‘ 

到 目前 为 止 ， 在 你 所 看 到 的 示例 中 ， 任 务 类 都 实现 了 Runnable。 在 非常 简单 的 情况 下 ， 你 
可 能 会 希望 使 用 直接 从 Thread 继 承 这 种 可 替换 的 方式 ， 就 像 下 面 这 样 : 


//: concurrency/SimpleThread, java 
// Inheriting directly from the Thread class. 


public class SimpleThread extends Thread { 
private int countDown = 5; 
private static int threadCount = 9; 
public SimpleThread() { 
// Store the thread name: 
super (Integer. toString (++threadCount)) ; 


start(); 
} 
public String toString() { 
return "#" + getName() + "(" + countDown + "), "; ‘ 
可 
public void run() { ? 
while(true) { 4 


System. out.print (this); 
if (--countDown == 6) 
return; 
} © 


public static void main(String{} args) { 
for(int 1 = @; i < 5: i++) 
new SimpleThread(); 
} 
} /* Output: 
#1(5), #1(4), #1(3), #1(2), #1(1), #2(5), #2(4), #2(3), 


#2(2), #2(1), #3(5), #3(4), #3(3), #3(2), #3(1), #4(5), 
#4(4), #4(3), #4(2), #4(1), #5(5), #5(4), #5(3), #5(2), 
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你 可 以 通过 调用 适当 的 Thread 构 造 器 为 Thread 对 象 赋予 具 体 的 名 称 ， 这 个 名 称 可 以 通过 使 
用 getName0 从 toString0 中 获得 。 
另 一 种 可 能 会 看 到 的 惯用 法 是 自 管理 的 Runnable: 


/1: concurrency/SelfManaged. java 
// A Runnable containing its own driver Thread. 


public class SelfManaged implements Runnable { 
private int countDown = 5; 
private Thread t = new Thread(this); 
public SelfManaged() { t.start(); } 
public String toString() { 
return Thread. currentThread().getName() + 
"(" + countDown + "), "; 
} 
public void run() { 
while(true) { TR 
System.out.print(this); 
if(--countDown == ©) 
return; 
} 


} 
public static void main(String[] args) { 
for(int i = @; 1 < 5; i++) 
new SelfManaged(); 

} 
) /* Output: 
Thread-@(5), Thread-0(4), Thread-6(3), Thread-@(2), Thread- 
(1), Thread-1(5), Thread-1(4), Thread-1(3), Thread-1(2), 
Thread-1(1), Thread-2(5), Thread-2(4), Thread-2(3), Thread- 
2(2), Thread-2(1), Thread-3(5), Thread-3(4), Thread-3(3), 
Thread-3(2), Thread-3(1), Thread-4(5), Thread-4(4), Thread- 
4(3), Thread-4(2), Thread-4(1), 
Whim 
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以 继承 另 一 个 不 同 的 类 ， 而 从 Thread 继 承 将 不 行 。 

注意 ，start0 是 在 构造 器 中 调用 的 。 这 个 示例 相当 简单 ， 因 此 可 能 是 安全 的 ， 但 是 你 应 该 意 
识 到 ， 在 构造 器 中 启动 线程 可 能 会 变 得 很 有 问题 ， 因 为 另 一 个 任务 可 能 会 在 构造 器 结束 之 前 开 
始 执行 ， 这 意味 着 该 任务 能 够 访问 处 于 不 稳定 状态 的 对 象 。 这 是 优选 Executor 而 不 是 显 式 地 创 
建 Thread 对 象 的 另 一 个 原因 。 

有 时 通过 使 用 内 部 类 来 将 线程 代码 隐藏 在 类 中 将 会 很 有 用 ， 就 像 下 面 这 样 : 


/1/: concurrency/ThreadVariations. java 
// Creating threads with inner classes. 
import java.util.concurrent.*; 

import static net.mindview.util.Print.*; 





// Using a named inner class: 
class InnerThreadl { 
private int countDown = 5; 
private Inner inner; 
private class Inner extends Thread { 
Inner (String name) { 
super (name) ; 
start(); 


} 
public void run() { P 
try { + 
while(true) { 
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print(this); 
if(--countDown == 0) return; 
steep(10); 
} 
} catch(InterruptedException e) { 
print("interrupted"); 
} 
} 
public String toString( 
return getName() + *: 
} 


{ 


+ countDown; 








} 
public InnerThread1(String name) { 
inner = new Inner (name); 
) 
} 


// Using an anonymous inner class: 
class InnerThread2 { 
private int countDown = 5; 
private Thread t: 
public InnerThread2(String name) { 
t = new Thread(name) { 
public void run() { 
try { 
while(true) { 
print (this); 
if(--countDown == 0) return: 
sleep(16) ; 
} 
} catch(InterruptedException e) { 
print("sleep() interrupted"); 
) 
} 
public String toString() { 
return getName() + ": ”+ countDown; 
) 
K 
t.start(); 
} 
} 


// Using a named Runnable implementation: 
class InnerRunnablel { 
private int countDown = 5; 
private Inner inner; 
private class Inner implements Runnable { 
Thread t; 
Inner (String name) { 
t = new Thread(this, name); 





t.start(); 
} 
public void run() { 
try { 
while(true) { 
print(this); 
if(--countDown == @) return; 
TimeUnit . MILLISECONDS. sleep(16) ; 
} 
} catch(Interruptedexception e) { 
print("sleep() interrupted"); 
} 
} 
public String toString() { 
return t.getName() + ": " + countDown; 
} 
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public InnerRunnablel (String name) { 
inner = new Inner (name) ; 
} 
} 


// Using an anonymous Runnable implementation: 
class InnerRunnable2 { 
private int countDown = 5; 
private Thread t; 
public InnerRunnable2(String name) { 
t = new Thread(new Runnable() { 
1139] public void run() { 
try { 
while(true) { 
print (this); 
if (--countDown == @) return; 
TimeUnit MILLISECONDS. sleep(10) ; 
} 
} catch(InterruptedException e) { 
print("sleep() interrupted"); 
} 
} 
public String toString() { 
return Thread.currentThread().getName() + 
"i * + countDown; 
} 


}, name); 
t.start(); 
$ 
} 




















// A separate method to run some code as a task: 
class ThreadMethod { 
private int countDown = 5; 
private Thread t; $ 
private String name; 
public ThreadMethod(String name) { this.name = name; } 
public void runTask() { 
if(t == null) { 
t = new Thread(name) { 
public void run() { 
try { 
while(true) { 
print(this); 
if(--countDown == 0) return; 
Sleep(10); 
} 
} catch(InterruptedException e) { 
print("sleep() interrupted"); 
} 
} . 
public String toString() { 
return getName() + ": "+ countDown: 
} 
}; 
1140] t.start(); 
} 
} 
public class ThreadVariations { 
public static void main(String{] args) { 
new InnerThreadl("InnerThread1") ; 
new InnerThread2("InnerThread2") ; 
new InnerRunnablel ("InnerRunnable1") ; 
new InnerRunnable2("InnerRunnable2") ; 
new ThreadMethod("ThreadMethod"). runTask() ; 
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$ A (Execute to see output) *///:~ 

InnerThread1 创 建 了 一 个 扩展 自 Thread 的 匿名 内 部 类 ， 并 且 在 构造 器 中 创建 了 这 个 内 部 类 
的 一 个 实例 。 如 果 内 部 类 具有 你 在 其 他 方法 中 需要 访问 的 特殊 能 力 (新 方法 )， 那 这 么 做 将 会 很 
有 意义 。 但 是 ， 在 大 多 数 时 候 ， 创 建 线程 的 原因 只 是 为 了 使 用 Thread 的 能 力 ， 因 此 不 必 创 建 匿 
名 内 部 类 。InnerThread2 展 示 了 可 替换 的 方式 : 在 构造 器 中 创建 一 个 匿名 的 Thread 子 类 ， 并 且 
将 其 向 上 转型 为 Thread 引 用 t。 如 果 类 中 的 其 他 方法 需要 访问 t， 那 它们 可 以 通过 Thread 接 口 来 
实现 ， 并 且 不 需要 了 解 该 对 象 的 确切 类 型 。 

该 示例 的 第 三 个 和 第 四 个 类 重复 了 前 面 的 两 个 类 ， 但 是 它们 使 用 的 是 Runnable 接 口 而 不 是 
Thread 类 。 

ThreadMethod 类 展示 了 在 方法 内 部 如 何 创建 线程 。 当 你 准备 好 运行 线程 时 ， 就 可 以 调用 这 
个 方法 ， 而 在 线程 开始 之 后 ， 该 方法 将 返回 。 如 果 该 线程 只 执行 辅助 操作 ， 而 不 是 该 类 的 重要 
操作 ， 那 么 这 与 在 该 类 的 构造 器 内 部 启动 线程 相 比 ， 可 能 是 一 种 更 加 有 用 而 适合 的 方式 。 

练习 10: (4) 按照 ThreadMethod 类 修改 练习 5， 使 得 runTask0 方 法 将 接受 一 个 参数 ， 表 示 要 
计算 总 和 的 斐 波 纳 契 数字 的 数量 ， 并 且 ， 每 次 调用 runTask0 时 ， 它 将 返回 对 submit0 的 调用 所 
产生 的 Future。 

21.2.10 术语 

正如 前 面 各 节 所 示 ， 在 Java 中 ， 你 可 以 选择 如 何 实现 并 发 编程 ， 并 且 这 个 选择 会 令 人 困惑 。 
这 个 问题 通常 来 自 于 用 来 描述 并 发 程序 技术 的 术语 ， 特 别 是 涉及 线程 的 那些 。 

到 目前 为 止 ， 你 应 该 看 到 要 执行 的 任务 与 驱动 它 的 线程 之 间 有 一 个 差异 ， 这 个 差异 在 Java 
类 库 中 尤为 明显 ， 因 为 你 对 Thread 类 实际 没有 任何 控制 权 (并 且 这 种 隔离 在 使 用 执行 器 时 更 加 
明显 ， 因 为 执行 器 将 替 你 处 理 线程 的 创建 和 管理 ) 。 你 创建 任务 ， 并 通过 某 种 方式 将 一 个 线程 附 
着 到 任务 上 ， 以 使 得 这 个 线程 可 以 驱动 任务 。 

在 Java 中 ，Thread 类 自身 不 执行 任何 操作 ， 它 只 是 驱动 赋予 它 的 任务 ， 但 是 线程 研究 中 总 是 
不 变 地 使 用 “线程 执行 这 项 或 那 项 动作 ”这 样 的 语言 。 因 此 ， 你 得 到 的 印象 就 是 “线程 就 是 任务 ”， 
当 我 第 一 次 碰 到 Java 线 程 时 ， 这 种 印象 非常 强烈 ， 以 至 于 我 看 到 了 一 种 明显 的 “是 一 个 ”关系 ， 
这 就 像 是 在 说 ， 很 明显 我 应 该 从 Thread 继 承 出 一 个 任务 。 另 外 ，Runnable 接 口 的 名 字 选 择 很 糟糕 ， 
所 以 我 认为 Task 应 该 是 好 得 多 名 字 。 如 果 接 口 只 是 其 方法 的 返 型 封装 ， 那 么 “ 它 执行 能 做 的 事情 ” 
这 种 命名 方式 将 是 恰当 的 ， 但 是 如 果 它 是 要 表示 更 高 层 的 抽象 ， 例 如 Task， 那 么 概念 名 将 有 用 。 

问题 是 各 种 抽象 级 别 被 混在 了 一 起 。 从 概念 上 讲 ， 我 们 希望 创建 独立 于 其 他 任务 运行 的 任 
务 ， 因 此 我 们 应 该 能 够 定义 任务 ， 然 后 说 “开始 "， 并 且 不 用 操心 其 细节 。 但 是 在 物理 上 ， 创 建 
线程 可 能 会 代价 高 昂 ， 因 此 你 必须 保存 并 管理 它们 。 这 样 ， 从 实现 的 角度 看 ， 将 任务 从 线程 中 
分 离 出 来 是 很 有 意义 的 。 另 外 ，Java 的 线程 机 制 基于 来 自 C 的 低级 的 p 线 程 方式 ， 这 是 一 种 你 必 
须 深入 研究 ， 并 且 需 要 须 完全 理解 其 所 有 事物 的 所 有 细节 的 方式 。 这 种 低级 特性 部 分 地 渗入 了 
Java 的 实现 中 ， 因 此 为 了 处 于 更 高 的 抽象 级 别 ， 在 编写 代码 时 ， 你 必须 遵循 规则 (我 将 在 本 章 
中 努力 演示 这 些 规则 )。 

为 了 澄清 这 些 讨论 ， 我 将 尝试 着 在 描述 将 要 执行 的 工作 时 使 用 术语 “任务 ”"， 只 有 在 我 引用 
到 驱动 任务 的 具体 机 制 时 ， 才 使 用 “线程 "。 因 此 ， 如 果 你 在 概念 级 别 上 讨论 系统 ， 那 就 可 以 只 
使 用 “任务 "， 而 压根 不 需要 提 及 驱动 机 制 。 

21.2.11 加 入 一 个 线程 
一 个 线程 可 以 在 其 他 线程 之 上 调用 join0 方 法 ， 其 效果 是 等 待 一 段 时 间 直 到 第 二 个 线程 结束 


1141 








[i143] 





670 种 21 章 





才 继续 执行 。 如 果 某 个 线程 在 另 一 个 线程 t 上 调用 tjoin0， 此 线程 将 被 挂 起 ， 直 到 目标 线程 t 结 束 
才 恢复 〈 即 tisAlive0 返 回 为 假 )。 

也 可 以 在 调用 join0 时 带 上 一 个 超时 参数 (单位 可 以 是 毫秒 ， 或 者 毫秒 和 纳 秒 ) ， 这 样 如 果 
程 在 这 段 时 间 到 期 时 还 没有 结束 的 话 ，join0 方 法 总 能 返回 。 

对 join0 方 法 的 调用 可 以 被 中 断 ， 做 法 是 在 调用 线程 上 调用 interrupt0 方 法 ， 这 时 需要 用 到 
try-catch 子 句 。 

下 面 这 个 例子 演示 了 所 有 这 些 操作 : 


//: concurrency/Joining. java 
// Understanding join(). 
import static net.mindview.util.Print.*; 





class Sleeper extends Thread { 
private int duration; 
public Sleeper(String name, int sleepTime) { 
super (name) ; 
duration = sleepTime; 
start); 


} 
public void run() { 
try { 
sleep(duration); 
} catch(InterruptedException e) { 
print(getName() + " was interrupted. ”+ 
“isInterrupted(): " + isInterrupted()); 
return: 


} 
print(getName() + " has awakened”); 


} 





i143] class Joiner extends Thread { 

private Sleeper sleeper; 

public Joiner(String name, Sleeper sleeper) { 
super (name) ; 
this.sleeper = sleeper; 
start(); 











y 
public void run() { 
try { 
sleeper. join(); 
} catch(Interruptedexception e) { 
print("Interrupted"): 


) 
print(getName() + ”join completed"); 
} 


public class Joining { 
public static void main(String{] args) { 
Sleeper 
sleepy = new Sleeper ("Sleepy", 1500), 
grumpy = new Sleeper ("Grumpy", 1568); 
Joiner 
dopey = new Joiner("Dopey", sleepy). 
doc = new Joiner("Doc", grumpy); 
grumpy. interrupt(): 





} 
} /* Output: 
Grumpy was interrupted. isInterrupted(): false 
Doc join completed 
Sleepy has awakened 
Dopey join completed 
Whim 
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Sleeper 是 一 个 Thread 类 型 ， 它 要 休眠 一 段 时 间 ， 这 段 时 间 是 通过 构造 器 传 进来 的 参数 所 指 
定 的 。 在 run0 中 ，sleep(0 方 法 有 可 能 在 指定 的 时 间 期 满 时 返回 ， 但 也 可 能 被 中 断 。 在 catch 子 句 
中 ， 将 根据 isInterrupted0 的 返回 值 报告 这 个 中 断 。 当 另 一 个 线程 在 该 线程 上 调用 interruptO 时 ， 
将 给 该 线程 设 定 一 个 标志 ， 表 明 该 线程 已 经 被 中 断 。 然 而 ， 异 常 被 捕获 时 将 清理 这 个 标志 ， 所 
以 在 catch 子 句 中 ， 在 异常 被 捕获 的 时 候 这 个 标志 总 是 为 假 。 除 异常 之 外 ， TTI 
他 情况 ， 比 如 线程 可 能 会 检查 其 中 断 状 态 。 

Joiner 线 程 将 通过 在 Sleeper 对 象 上 调用 join 方法 来 等 待 Sleeper 醒 来 。 在 main0 里 面 ， 每 个 
Sleeper 都 有 一 个 Joiner， 这 可 以 在 输出 中 发 现 ， 如 果 Sleeper 被 中 断 或 者 是 正常 结束 ，Joiner 将 
和 Sleeper 一 同 结束 。 

YER, Java SE5 的 java.util.concurrent 类 库 包 含 诸如 CyclicBarrier (本 章 稍 后 会 展示 ) 这 样 
的 工具 ， 它 们 可 能 比 最 初 的 线程 类 库 中 的 join0 更 加 适合 。 

21.2.12 创建 有 响应 的 用 户 界面 

如 前 所 述 ， 使 用 线程 的 动机 之 一 就 是 建立 有 响应 的 用 户 界面 。 尽 管 我 们 要 到 第 22 章 才 接触 
到 图 形 用 户 界面 ， 但 下 面 还 是 给 出 了 一 个 基于 控制 台 用 户 界面 的 简单 教学 示例 。 下 面 的 例子 有 
两 个 版 本 : 一 个 关注 于 运算 ， 所 以 不 能 读 取 控制 台 输入 ， 另 一 个 把 运算 放 在 任务 里 单独 运行 ， 
此 时 就 可 以 在 进行 运算 的 同时 监听 控制 台 输 入 。 


1/: concurréncy/ResponstveUl. java 
// User interface responsiveness. 
77 {RunByHand} 


class UnresponsiveUI { 
private volatile double d = 1; 
Public UnresponsiveUI() throws Exception { $ 
while(d > 0) 
d =d + (Math.PI + Math.E) / d; 
System. in.read(); // Never gets here 


} 


public class ResponsiveUI extends Thread { 
private static volatile double d = 1; 
public ResponsiveUI() { 
setDaemon(true) ; 
start(); 


public void run() { 
while(true) { 
d = d + (Math.PI + Math.E) / d; 
} 
了 
public static void main(String[] args) throws Exception { 
//! new UnresponsiveUI(); // Must kill this process 
new ResponsiveUI(); 
System. in.read(); 
System.out.printin(d); // Shows progress 


} 
} Mi~ 
UnresponsiveUI 在 一 个 无 限 的 while 循 环 里 执行 运算 ， 显 然 程序 不 可 能 到 达 读 取 控制 台 输入 
的 那 一 行 《编译 器 被 欺骗 了 ， 相 信 while 的 条 件 使 得 程序 能 到 达 读 取 控制 台 输入 的 那 一 行 )。 如 
果 把 建立 UnresponsiveUI 的 那 一 行 的 注释 解除 掉 再 运行 程序 ， 那 么 要 终止 它 的 话 ， 就 只 能 杀 死 
这 个 进程 。 
要 想 让 程序 有 响应 ， 就 得 把 计算 程序 放 在 run0 方 法 中 ， 这 样 它 就 能 让 出 处 理 器 给 别 的 程序 。 
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当 你 按 下 “ 回 车 ” 键 的 时 候 ， 可 以 看 到 计算 确实 在 作为 后 台 程序 运行 ， 同 时 还 在 等 待 用 户 输入 。 


21.2.13 线程 组 

线程 组 持 有 一 个 线程 集合 。 线 程 组 的 价值 可 以 引用 Joshua Bloch 的 话 来 总 结 ， 他 在 Sun 时 是 
软件 架构 师 ， 订 正 并 极 大 地 改善 了 JDK1.2 中 Java 集 合 类 库 ; , 

“最 好 把 线程 组 看 成 是 一 次 不 成 功 的 尝试 ， 你 只 要 忽略 它 就 好 了 。” 

如 果 你 花费 了 大 量 的 时 间 和 精力 试图 发 现 线程 组 的 价值 (就 像 我 一 样 )， 那 么 你 可 能 会 惊异 ， 
为 什么 没有 来 自 Sun 的 关于 这 个 主题 的 官方 声明 ， 多 年 以 来 ， 相 同 的 问题 对 于 Java 发 生 的 其 他 变 
化 也 询问 过 无 数 遍 。 诺 贝尔 经 济 学 奖 得 主 Joseph Stiglitz 的 生活 哲学 可 以 用 来 解释 这 个 问题 8 ， 它 
被 称 为 承诺 升级 理论 (The Theory of Escalating Commitment) ; 

“继续 错误 的 代价 由 别人 来 承担 ， 而 承认 错误 的 代价 由 自己 承担 ;” 


21.2.14 捕获 异常 

由 于 线程 的 本 质 特性 ， 使 得 你 不 能 捕获 从 线程 中 逃逸 的 异常 。 一 旦 异常 逃 出 任务 的 ran( 方 
法 ， 它 就 会 向 外 传播 到 控制 台 ， 除 非 你 采取 特殊 的 步骤 捕获 这 种 错误 的 异常 。 在 Java SE5 之 前 ， 
你 可 以 使 用 线程 组 来 捕获 这 些 异 常 ， 但 是 有 了 Java SE5， 就 可 以 用 Executor 来 解决 这 个 问题 ， 
此 你 就 不 再 需要 了 解 有 关 线 程 组 的 任何 知识 了 (除非 要 理解 遗留 代码 ， 请 查看 可 以 从 www. 
MindView.net 下 载 的 《Thinking in Java (2nd Edition)》， 以 了 解 线程 组 的 细节 ) 。 

下 面 的 任务 总 是 会 抛 出 一 个 异常 ， 该 异常 会 传播 到 其 run0 方 法 的 外 部 ， 并 且 main0 展 示 了 
当 你 运行 它 时 所 发 生 的 事情 : 


//: concurrency/Except ionThread. java 
// {ThrowsException} 
import java.util.concurrent.*: 


public class ExceptionThread implements Runnable { 
public void run() { 
throw new RuntimeException() ; 


} 

public static void main(String(] args) { 
ExecutorService exec = Executors.newCachedThreadPool() ; 
exec.execute(new ExceptionThread()); 


} 
} Mi~ 
输出 如 下 (将 某 些 限定 符 修整 为 适合 显示 ) : 


java. lang. RuntimeException 
at ExceptionThread. run(ExceptionThread.java:7) 
at ThreadPoolExecutor$Worker.runTask (Unknown Source) 
at ThreadPoolExecutor$Worker.run (Unknown Source) 
at java. 1lang.Thread.run(Unknown Source) 


将 main 的 主体 放 到 try-catch 语 句 块 中 是 没有 作用 的 : 


71 concurrency/NaiveExceptionHandling. java 
1/ {ThrowsException} 
import java.util.concurrent.*; 
public class NaiveExceptionHandling { 
public static void main(String[] args) { 
try { 
ExecutorService exec = 
Executors.newCachedThreadPool () ; 
exec.execute(new ExceptionThread()); 


© (Effective Java™ Programming Language Guide), Joshua Bloch#, Addison-Wesley, 2001, 第 211 页 。 本 书 中 文 版 
已 由 机 械 工业 出 版 社 出 版 。 一 一 编辑 注 
O 以 及 贯穿 于 Java 有 生 以 来 的 其 他 许多 问题 。 嘿 , 问 什么 止步 于 此 呢 ? 因 为 我 已 经 参考 过 很 多 存在 这 个 问题 的 项 目 了 。 
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} catch(RuntimeException ue) { 
// This statement will NOT execute! 
System. out .printtn("Exception has been handlet 


} 

} /AL h 
这 将 产生 与 前 面 示例 相同 的 结果 : 未 捕获 的 异常 。 

为 了 解决 这 个 问题 ， 我 们 要 修改 Executor 产 生 线程 的 方式 。Thread.UncaughtException- 
Handler 是 Java SE5 中 的 新 接口 ， 它 允许 你 在 每 个 Thread 对 象 上 都 附着 一 个 异常 处 理 器 。 
Thread.UncaughtExceptionHandler.uncaughtException0 会 在 线程 因 未 捕获 的 异常 而 临近 死亡 时 
被 调用 。 为 了 使 用 它 ， 我 们 创建 了 一 个 新 类 型 的 ThreadFactory， 它 将 在 每 个 新 创建 的 Thread 对 
象 上 附着 一 个 Thread.UncaughtExceptionHandler。 我 们 将 这 个 工厂 传递 给 Executors 创 建新 的 
ExecutorService 的 方法 : 


//: concurrency/CaptureUncaughtException.java 
import java.util.concurrent.*; 





class ExceptionThread2 implements Runnable { : 
public void run() { F 
Thread t = Thread. currentThread() ; 
System.out.printin("run() by ”+ t); 
System.out.printtn( 
"eh = ”+ t.getUncaughtExceptionHandler()): 
throw new RuntimeException(); 
} 
} 


class MyUncaughtExceptionHandler implements 
Thréad.UncaughtExceptionHandler { 
public void uncaughtException(Thread t, Throwable e) { 
System, out,printin("caught " + e); 


} 


class HandlerThreadFactory implements ThreadFactory { 
public Thread newThread(Runnable r) { 
System.out.printin(this + " creating new Thread"); 
Thread t = new Thread(r); 
System.out.printin("created " + t); 
t.setUncaughtExcept ionHandler ( 
new MyUncaughtExcept ionHandler()); 
System. out. printin( 
"eh = ”+ t.getUncaughtExcept ionHandler()); 
return t; 
} 


} 


public class CaptureUncaughtException { 
public static void main(String[} args) { F 
ExecutorService exec = Executors.newCachedThreadPool ( 
new HandlerThreadFactory()); 
exec.execute(new Except ionThread2()) ; 


} 
} /* Output: (96% match) 
HandlerThreadFactory@de6ced creating new Thread 
created Thread(Thread-,5,main] 
eh = MyUncaughtExceptionHandler@1fb8ee3 
run() by Thread{Thread-@,5.main} 
eh = MyUncaughitExcept ionHandler@1fb8ee3 3 
caught java. lang.Runt imeException 
T= 


在 程序 中 添加 了 额外 的 跟踪 机 制 ， 用 来 验证 工厂 创建 的 线程 会 传递 给 UncaughtException- 
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Handler。 你 现在 可 以 看 到 ， 未 捕获 的 异常 是 通过 uncaughtException 来 捕获 的 。 

上 面 的 示例 使 得 你 可 以 按照 具体 情况 逐个 地 设置 处 理 器 。 如 果 你 知道 将 要 在 代码 中 处 处 使 
用 相同 的 异常 处 理 器 ， 那 么 更 简单 的 方式 是 在 Thread 类 中 设置 一 个 静态 域 ， maama 
置 为 默认 的 未 捕获 异常 处 理 器 : 


//: concurrency/SettingDefaultHandler. java 
import java.util.concurrent.*; 


public class SettingDefaultHandler ¢ 
public static void main(String[] args) { 
1 Thread. setDefaul tUncaughtExceptionHandler( 
new HyUncaughtExceptionHandler()); 
ExecutorService exec = Executors.newCachedThreadPool(); 
exec. execute (new ExceptionThread()): 
} 
} /* Output: 
caught java. lang.RuntimeException 


Wh f 


这 个 处 理 器 只 有 在 不 存在 线程 专 有 的 未 捕获 异常 处 理 器 的 情况 下 才 会 被 调用 。 系 统 会 检查 
线程 专 有 版 本 ， 如 果 没 有 发 现 ， 则 检查 线程 组 是 否 有 其 专 有 的 uncaughtException0 方 法 ， 如 果 
也 没有 ， 再 调用 defaultUncaughtExceptionHandler。 


21.3 共享 受 限 资源 


可 以 把 单线 程 程序 当 作 在 问题 域 求解 的 单一 实体 ， 每 次 只 能 做 一 件 事情 。 因 为 只 有 一 个 实 
体 ， 所 以 永远 不 用 担心 诸如 “两 个 实体 试图 同时 使 用 同一 个 资源 ”这 样 的 问题 -比如 ， 两 个 人 在 
同一 个 地 方 停车 ， 两 个 人 同时 走 过 一 扇 门 ， 甚 至 是 两 个 人 同时 说 话 。 

有 了 并 发 就 可 以 同时 做 多 件 事 情 了 ， 但 是 ， 两 个 或 多 个 线程 彼此 互相 干涉 的 问题 也 就 出 现 
了 。 如 果 不 防 范 这 种 冲突 ， 就 可 能 发 生 两 个 线程 同时 试图 访问 同一 个 银行 账户 ， 或 向 同一 个 打 
印 机 打印 ， 改 变 同一 个 值 等 诸如 此 类 的 问题 。 

21.3.1 不 正确 地 访问 资源 

考虑 下 面 的 例子 ， 其 中 一 个 任务 产生 偶数 ， 而 其 他 任务 消费 这 些 数字 。 这 里 ， 消 费 者 任务 
的 唯一 工作 就 是 检查 偶数 的 有 效 性 。 

首先 ， 我 们 定义 EvenChecker， 即 消费 者 任务 ， 因 为 它 将 在 随后 所 有 的 示例 中 被 复 用 。 为 
了 将 EvenChecker 与 我 们 要 试验 的 各 种 类 型 的 生成 器 解 耦 ， 我 们 将 创建 一 个 名 为 IntGenerator 的 
抽象 类 ， 它 包含 EvenChecker 必 须 了 解 的 必 不 可 少 的 方法 ， 即 一 个 next0 方 法 ， 和 一 个 可 以 执行 
撤销 的 方法 。 这 个 类 没有 实现 Generator 接 口 ， 因 为 它 必须 产生 一 个 int， 而 泛 型 不 支持 基本 类 型 
的 参数 ， 

41: concurrency/IntGenerator.java 


public abstract class IntGenerator { 
private volatile boolean canceled = false: 
public abstract int next(); 
// Allow this to be canceled: 
public void cancel() { canceled = trite; } 
public boolean isCanceled() { return canceled; } 
} Ii~ 


IntGenerator 有 一 个 cancel0 方 法 ， 可 以 修改 boolean 类 型 的 canceled 标 志 的 状态 ， 还 有 一 个 
isCanceled0 方 法 ， 可 以 查看 该 对 象 是 否 已 经 被 取消 。 因 为 canceled 标 志 是 boolean 类 型 的 ， 所 以 
它 是 原子 性 的 ， 即 诸如 赋值 和 返回 值 这 样 的 简单 操作 在 发 生 时 没有 中 断 的 可 能 ， 因 此 你 不 会 看 
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到 这 个 域 处 于 在 执行 这 些 简单 操作 的 过 程 中 的 中 间 状 态 。 为 了 保证 可 视 性 ，canceled 标 志 还 是 
volatile 的 。 你 将 在 本 章 稍 后 学 习 原 子 性 和 可 视 性 。 
任何 IntGenerator 都 可 以 用 下 面 的 EvenChecker 类 来 测试 : 


//: concurrency/EvenChecker . java 
import java.util.concurrent.*; 


public class EvenChecker implements Runnable { 
private IntGenerator generator; 
private final int id; 
public EvenChecker(IntGenerator g, int ident) { 
generator = g; 
id = ident; 
} 
public void run() { 
. while(!generator. isCanceled()) { 
int val = generator.next(); 
if(val % 2 != 6) { 
System.out.printin(val + " not even!"); 
generator.cancel(); // Cancels all EvenCheckers 
} 
} 
$ 
// Test any type of IntGenerator: 
public static void test(IntGenerator gp. int count) { 
System.out.printin("Press Control-C to exit"); 
ExecutorService exec = Executors.newCachedThreadPool(); 
for(int i = 0; i < count; i++) 
exec.execute(new EvenChecker(gp, i)): 
exec. shutdown() ; 





} 

// Default value for count: 

public static void test(IntGenerator gp) { 
test(gp, 10); 


) 

YM 

注意 ， 在 本 例 中 可 以 被 撤销 的 类 不 是 Runnable， 而 所 有 依赖 于 IntGenerator 对 象 的 
EvenChecker 任 务 将 测试 它 ， 以 查看 它 是 否 已 经 被 撤销 ， 正 如 你 在 run0 中 所 见 。 通 过 这 种 方式 ， 
共享 公共 资源 (IntGenerator) 的 任务 可 以 观察 该 资源 的 终止 信号 。 这 可 以 消除 所 谓 竞争 条 件 ， 
即 两 个 或 更 多 的 任务 竞争 响应 某 个 条 件 ， 因 此 产生 冲突 或 不 一 致 结果 的 情况 。 你 必须 仔细 考虑 
并 防范 并 发 系统 失败 的 所 有 可 能 途径 ， 例 如 ， 一 个 任务 不 能 依赖 于 另 一 个 任务 ， 因 为 任务 关闭 
的 顺序 无 法 得 到 保证 。 这 里 ， 通 过 使 任务 依赖 于 非 任务 对 象 ， 我 们 可 以 消除 潜在 的 竞争 条 件 。 

test0 方 法 通过 启动 大 量 使 用 相同 的 IntGenerator 的 EvenCliecker， 设 置 并 执行 对 任何 类 型 的 
IntGenerator 的 测试 。 如 果 IntGenerator 引 发 失败 ， 那 么 test0 将 报告 它 并 返回 ， 否 则 ， 你 必须 按 
下 Control-C 来 终止 它 。 

EvenChecker 任 务 总 是 读 取 和 测试 从 与 其 相关 的 IntGenerator 返 回 的 值 。 注 意 ， 如 果 
generator.isCanceled0 为 true， 则 run0 将 返回 ， 这 将 告知 EvenChecker.test0 中 的 Executor 该 任务 
完成 了 。 任 何 EvenChecker 任 务 都 可 以 在 与 其 相关 联 的 IntGenerator 上 调用 cancel0， 这 将 导致 
所 有 其 他 使 用 该 IntGenerator 的 EvenChecker 得 体 地 关闭 。 在 后 面 各 节 中 ， 你 将 看 到 Java 包 含 的 
用 于 线程 终止 的 各 种 更 通用 的 机 制 。 

我 们 看 到 的 第 一 个 IntGenerator 有 一 个 可 以 产生 一 系列 偶数 值 的 next0 方 法 : 


//: concurrency/EvenGenerator.java 
// When threads collide. 


public class EvenGenerator extends IntGenerator { 
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private int currentEvenValue = 0; 
public int next() { 
+*currentEvenValue: // Danger point here! 
++currentEvenValue; ` 
return currentEvenValue: 
} * 1 
Public static void main(String[] args) { > 
EvenChecker. test (new EvenGenerator()); 


ri 
} /* Output: (Sample) 
Press Control-C to exit 
89476993 not even! 
89476993 not even! 
“i~ 


一 个 任务 有 可 能 在 另 一 个 任务 执行 第 一 个 对 currentEvenValue 的 递增 操作 之 后 ， 但 是 没有 
执行 第 二 个 操作 之 前 ， 调 用 next0 方 法 ( 即 ， 代 码 中 被 注释 为 “Danger point here!” 的 地 方 )。 这 
将 使 这 个 值 处 于 “不 恰当 ”的 状态 。 为 了 证 明 这 是 可 能 发 生 的 ，EvenChecker.test0 创 建 了 一 组 
EvenChecker 对 象 ， 以 连续 地 读 取 并 输出 同一 个 EvenGenerator， 并 测试 检查 每 个 数值 是 否 都 是 
偶数 。 如 果 不 是 ， 就 会 报告 错误 ， 而 程序 也 将 关闭 。 

这 个 程序 最 终 将 失败 ， 因 为 各 个 EvenChecker 任 务 在 EvenGenerator 处 于 “不 恰当 的 ”状态 
时 ， 仍 能 够 访问 其 中 的 信息 。 但 是 ， 根 据 你 使 用 的 特定 的 操作 系统 和 其 他 实现 细节 ， 直 到 
EvenCenerator 完 成 多 次 循环 之 前 ， 这 个 问题 都 不 会 被 探测 到 。 如 果 你 希望 更 快 地 发 现 失败 ， 可 
以 尝试 着 将 对 yield0 的 调用 放置 到 第 一 个 和 第 二 个 递增 操作 之 间 。 这 只 是 并 发 程序 的 部 分 问 
题 一 如果 失败 的 概率 非常 低 ， 那 么 即使 存在 缺陷 ,它们 也 可 能 看 起 来 是 正确 的 。 

有 一 点 很 重要 ， 那 就 是 要 注意 到 递增 程序 自身 也 需要 多 个 步 又 ， 并 且 在 递增 过 程 中 任务 可 
能 会 被 线程 机 制 挂 起 一 一 也 就 是 说 ， 在 Java 中 ， 递增 不 是 原子 性 的 操作 。 因 此 ， 如 果 不 保护 任务 ， 
即使 单一 的 递增 也 不 是 安全 的 。 

21.3.2 解决 共享 资源 竞争 

前 面 的 示例 展示 了 使 用 线程 时 的 一 个 基本 问题 ， 你 永远 都 不 知道 一 个 线程 何 时 在 运行 。 想 
象 一 下 ， 你 坐 在 桌 边 手 拿 又 子 ， 正 要 去 又 盘 子 中 的 最 后 一 片 食物 ， 当 你 的 又 子 就 要 够 着 它 时 ， 
这 片 食物 突然 消失 了 ，` 因 为 你 的 线程 被 挂 起 了 ， 而 另 一 个 餐 者 进入 并 吃 掉 了 它 。 这 正 是 在 你 编 
写 并 发 程序 时 需要 处 理 的 问题 。 对 于 并 发 工作 ， 你 需要 某 种 方式 来 防止 两 个 任务 访问 相同 的 次 
源 ， 至 少 在 关键 阶段 不 能 出 现 这 种 情况 。 

防止 这 种 冲突 的 方法 就 是 当 资 源 被 一 个 任务 使 用 时 ， 在 其 上 加 锁 。 第 一 个 访问 某 项 资源 的 
任务 必须 锁定 这 项 资源 ， 使 其 他 任务 在 其 被 解锁 之 前 ， 就 无 法 访问 它 了 ， 而 在 其 被 解锁 之 时 ， 
另 一 个 任务 就 可 以 锁定 并 使 用 它 ， 以 此 类 推 。 如 果 汽 车 前 排 座位 是 受 限 资 源 ， 那 么 大 喊 着 “ 冲 
呀 ! ”的 孩子 就 会 (在 这 次 旅途 过 程 中 ) 获取 其 上 的 锁 。 

基本 上 所 有 的 并 发 模式 在 解决 线程 冲突 问题 的 时 候 ， 都 是 采用 序列 化 访问 共享 资源 的 方案 。 
这 意味 着 在 给 定时 刻 只 允许 一 个 任务 访问 共享 资源 。 通 常 这 是 通过 在 代码 前 面 加 上 一 条 锁 语句 ， 
来 实现 的 ， 这 就 使 得 在 一 段 时 间 内 只 有 一 个 任务 可 以 运行 这 段 代 码 。 因 为 锁 语句 产生 了 一 种 互 
相 排斥 的 效果 ， 所 以 这 种 机 制 常常 称 为 互 斤 量 (mutex) 。 \ ; 

考虑 一 下 屋子 里 的 浴室 ; 多 个 人 〈 即 多 个 由 线程 驱动 的 任务 ) 都 希望 能 单独 使 用 浴室 (BD 
共享 资源 )。 为 了 使 用 浴室 ， 一 个 人 先 敲 门 ， 看 看 是 否 能 使 用 。 如 果 没 人 的 话 ， 他 就 进入 浴室 并 
锁 上 门 。 这 时 其 他 人 要 使 用 浴室 的 话 ， 就 会 被 “阻挡 "， 所 以 他 们 要 在 浴室 门口 等 待 ， 直 到 浴室 
可 以 使 用 。 

当 滩 室 使 用 完毕 ， 就 该 把 浴室 给 其 他 人 使 用 了 ( 别 的 任务 就 可 以 访问 资源 了 )， 这 个 比喻 就 





HR 677 





有 点 不 太 准 确 了 。 事 实 上 ， 人 们 并 没有 排队 ， 我 们 也 不 能 确定 谁 将 是 下 一 个 使 用 浴室 的 人 ， 
为 线程 调度 机 制 并 不 是 确定 性 的 。 实 际 情况 是 : 等 待 使 用 浴室 的 人 们 簇拥 在 浴室 门口 ， 当 锁 住 
浴室 门 的 那个 人 打开 锁 准 备 离开 的 时 候 ， 离 门 最 近 的 那个 人 可 能 进入 浴室 。 如 前 所 述 ， 可 以 通 
过 yield0 和 setPriority0 来 给 线程 调度 器 提供 建议 ， 但 这 些 建议 未 必 会 有 多 大 效果 ， 这 取决 于 你 
的 具体 平台 和 JVM 实 现 。 

Java 以 提供 关键 字 synchronized 的 形式 ， 为 防止 资源 冲突 提供 了 内 置 支持 。 当 任务 要 执行 被 
synchronized 关 键 字 保护 的 代码 片段 的 时 候 ， 它 将 检查 锁 是 否 可 用 ， 然 后 获取 锁 ， 执 行 代码 ， 释 
放 锁 。 G 

共享 资源 一 般 是 以 对 象形 式 存在 的 内 存 片段 ， 但 也 可 以 是 文件 、 输 入 /输出 端口 ， 或 者 是 打 
印 机 。 要 控制 对 共享 资源 的 访问 ， 得 先 把 它 包 装 进 一 个 对 象 。 然 后 把 所 有 要 访问 这 个 资源 的 方法 
标记 为 synchronized。 如 果 某 个 任务 处 于 一 个 对 标记 为 synchronized 的 方法 的 调用 中 ， 那 么 在 这 
个 线程 从 该 方法 返回 之 前 ， 其 他 所 有 要 调用 类 中 在 何 标记 为 synchronized 方 法 的 线程 都 会 被 阻塞 。 

在 生成 偶数 的 代码 中 ， 你 已 经 看 到 了 ， 你 应 该 将 类 的 数据 成 员 都 声明 为 private 的 ， 而 且 只 
能 通过 方法 来 访问 这 些 数据 ， 所 以 可 以 把 方法 标记 为 synchronized 来 防止 资源 冲突 。 下 面 是 声明 
synchronized 方 法 的 方式 : 


synchronized void f(y) { /* ... */} i 
synchronized void g() { /* ... */ } 


所 有 对 象 都 自动 含有 单一 的 锁 〈 也 称 为 监视 器 ) 。 当 在 对 象 上 调用 其 任意 synchronized 方 法 
的 时 候 ， 此 对 象 都 被 加 锁 ， 这 时 该 对 象 上 的 其 他 synchronized 方 法 只 有 等 到 前 一 个 方法 调用 完毕 
并 释放 了 锁 之 后 才能 被 调用 。 对 于 前 面 的 方法 ， 如 果 某 个 任务 对 对 象 调 用 了 f0， 对 于 同一 个 对 
象 而 言 ， 就 只 能 等 到 (0 调用 结束 并 释放 了 锁 之 后 ， 其 他 任务 才能 调用 (0 和 gO。 所 以 ， 对 于 某 个 
特定 对 象 来 说 ， 其 所 有 synchronized 方 法 共享 同一 个 锁 ， 这 可 以 被 用 来 防止 多 个 任务 同时 访问 被 
编码 为 对 象 内 存 。 

注意 ， 在 使 用 并 发 时 ， 将 域 设置 为 private 是 非常 重要 的 ， 否 则 ，synchronized 关 键 字 就 不 能 
防止 其 他 任务 直接 访问 域 ， 这 样 就 会 产生 冲突 。 

一 个 任务 可 以 多 次 获得 对 象 的 锁 。 如 果 一 个 方法 在 同一 个 对 象 上 调用 了 第 二 个 方法 ， 后 者 
又 调用 了 同一 对 象 上 的 另 一 个 方法 ， 就 会 发 生 这 种 情况 。JVM 负 责 跟踪 对 象 被 加 锁 的 次 数 。 如 
果 一 个 对 象 被 解锁 〈 即 锁 被 完全 释放 ) ， 其 计数 变 为 0。 在 任务 第 一 次 给 对 象 加 锁 的 时 候 ， 计 数 
变 为 1。 每 当 这 个 相同 的 任务 在 这 个 对 象 上 获得 锁 时 ， 计 数 都 会 递增 。 显 然 ， 只 有 首先 获得 了 锁 
的 任务 才能 人 允许 继续 获取 多 个 锁 。 每 当 任 务 离开 一 个 synchronized 方 法 ， 计 数 递 减 ， 当 计数 为 堆 
的 时 候 ， 锁 被 完全 释放 ， 此 时 别 的 任务 就 可 以 使 用 此 资源 。 

针对 每 个 类 ， 也 有 一 个 锁 (作为 类 的 Class 对 象 的 一 部 分 )}， 所 以 synchronized static 方 法 可 
以 在 类 的 范围 内 防止 对 static 数 据 的 并 发 访问 。 

你 应 该 什么 时 候 同 步 呢 ? 可 以 运用 Brian 的 同步 规则 

如 果 你 正在 写 一 个 变量 ， 它 可 能 接 下 来 将 被 另 一 个 线程 读 取 ， 或 者 正在 读 取 一 个 上 一 次 已 经 
被 另 一 个 线程 写 过 的 变量 ， 那 么 你 必须 使 用 同步 ， 并 且 ， 读 写 线程 都 必须 用 相同 的 监视 器 锁 同步 。 

如 果 在 你 的 类 中 有 超过 一 个 方法 在 处 理 临界 数据 ， 那 么 你 必须 同步 所 有 相关 的 方法 。 如 果 只 
同步 一 个 方法 ， 那 么 其 他 方法 将 会 随意 地 名 略 这 个 对 象 锁 ， 并 可 以 在 无 任何 惩罚 的 情况 下 被 调用 。 
这 是 很 重要 的 一 点 : 每 个 访问 临界 共享 资源 的 方法 都 必须 被 同步 ， 否 则 它们 就 不 会 正确 地 工作 。 


© 引 自 Brian Goetz, (Java Concurrency in Practice》 的 作者 ， 这 本 书 的 作者 包括 Brian Goetz, Tim Peierls, Joshua 
Bloch, Joseph Bowbeer, David Holmes 和 Doug Lea (Addison-Wesley, 2006), 
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同步 控制 EvenGenerator 
通过 在 EvenGeneratorjava 中 加 入 synchronized 关 键 字 ， 可 以 防止 不 希望 的 线程 访问 ; 


JJ: concurrency/SynchronizedEvenGenerator .java 
// Simplifying mutexes with the synchronized keyword. 
+7 {RunByHand} 


public class 
SynchronizedEvenGenerator extends IntGenerator { 
private int currentEvenValue = @; 
public syrichronized int next() { 
++currentEvenValue; 
Thread.yield(); // Cause failure faster 
++currentEvenValue; 
return currentEvenValue; 


} 
public static void main(String[] args) { 
EvenChecker .test(new SynchronizedEvenGenerator()); 


} H~ 

对 Thread.yield0 的 调用 被 插入 到 了 两 个 递增 操作 之 间 ， 以 提高 在 currehtEvenValue 是 奇数 
状态 时 上 下 文 切 换 的 可 能 性 。 因 为 互 斥 可 以 防止 多 个 任务 同时 进入 临界 区 ， 所 以 这 不 会 产生 任 
何 失败 。 但 是 如 果 失 败 将 会 发 生 ， 调 用 yield0 是 一 种 促使 其 发 生 的 有 效 方式 。 

第 一 个 进入 next0 的 任务 将 获得 锁 , 任何 其 他 试图 获取 锁 的 任务 都 将 从 其 开始 尝试 之 时 被 阻塞 ， 
直至 第 一 个 任务 释放 锁 。 通 过 这 种 方式 ， 任 何 时 刻 只 有 一 个 任务 可 以 通过 由 互 斥 量 看 护 的 代码 。 

练习 11，(3) 创建 一 个 类 ， 它 包含 两 个 数据 域 和 一 个 操作 这 些 域 的 方法 ， 其 操作 过 程 是 多 步 
附 的 。 这 样 在 该 方法 执行 过 程 中 ， 这 些 域 将 处 于 “不 正确 的 状态 ”( 根 据 你 设 定 的 某 些 定义 )。 
添加 读 取 这 些 域 的 方法 ， 创 建 多 个 线程 去 调用 各 种 方法 ， 并 展示 处 于 “不 正确 状态 的 ”数据 是 
可 视 的 。 使 用 synchronized 关 键 字 修复 这 个 问题 。 

使 用 显 式 的 Lock 对 象 

Java SE5 的 java.util.concurrent 类 库 还 包含 有 定义 在 java.util.concurrent.locks 中 的 显 式 的 互 
斥 机 制 。Lock 对 象 必须 被 显 式 地 创建 、 锁 定 和 释放 。 因 此 ， 它 与 内 建 的 锁 形 式 相 比 ， 代 码 缺 乏 
优雅 性 。 但 是 ， 对 于 解决 某 些 类 型 的 问题 来 说 ， 它 更 加 灵活 。 下 面 用 显 式 的 Lock 重 写 的 是 
SyschronizedEventGenerator.java: 


//: concurrency/MutexEvenGenerator.java 
// Preventing thread collisions with mutexes. 
/1/ {RunByHand) 

import java.util.concurrent.locks.*; 


public class MutexEvenGenerator extends IntGenerator { 
private int currentEvenValue = @; » 
private Lock lock = new ReentrantLock(); 
public int next() { 
lock. Lock() ; 
try { 
++currentEvenValue; 
Thread.yield(); // Cause failure faster 
++currentEvenValue; 
return currentEvenValue: 
} finally { 
lock. unlock () # 
} 


} 
public static void main(String{] args) { 
EvenChecker.test(new MutexEvenGenerator()): 


} 
} Mi~ 
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MutexEvenGenerator 添 加 了 一 个 被 互 斥 调用 的 锁 ， 并 使 用 lock0 和 unlock(0 方 法 在 next0 内 
部 创建 了 临界 资源 。 当 你 在 使 用 Lock 对 象 时 ， 将 这 里 所 示 的 惯用 法 内 部 化 是 很 重要 的 : 紧 接着 
的 对 lock0 的 调用 ， 你 必须 放置 在 finally 子 句 中 带 有 unlock0 的 try-finally 语 句 中 。 注 意 ，return 
语句 必须 在 try 子 句 中 出 现 ， 以 确保 unlock0 不 会 过 早 发 生 ， 从 而 将 数据 暴露 给 了 第 二 个 任务 。 

尽管 try-finally 所 需 的 代码 比 synchronized 关 键 字 要 多 ， 但 是 这 也 代表 了 显 式 的 Lock 对 象 的 
优点 之 一 。 如 果 在 使 用 synchronized 关 键 字 时 ， 某 些 事物 失败 了 ， 那 么 就 会 抛 出 一 个 异常 。 但 是 
你 没有 机 会 去 做 任何 清理 工作 ， 以 维护 系统 使 其 处 于 良好 状态 。 有 了 显 式 的 Lock 对 象 ， 你 就 可 
以 使 用 finally 子 句 将 系统 维护 在 正确 的 状态 了 。 

大 体 上 ， 当 你 使 用 synchronized 关 键 字 时 ， 需 要 写 的 代码 量 更 少 ， 并 且 用 户 错误 出 现 的 可 能 
性 也 会 降低 ， 因 此 通常 只 有 在 解决 特殊 问题 时 ， 才 使 用 显 式 的 Lock 对 象 。 例 如 ， 用 synchronized 
关键 字 不 能 尝试 着 获取 锁 且 最 终 获取 锁 会 失败 ， 或 者 尝试 着 获取 锁 一 段 时 间 ， 然 后 放弃 它 ， 要 
实现 这 些 ， 你 必须 使 用 concurrent 类 库 ; 


//: concurrency/AttemptLocking. java 

// Locks in the concurrent library allow you 
// to give up on trying to acquire a lock. 
import java.util.concurrent.*; 

import java.util.concurrent.locks.*; 


public class AttemptLocking { 
private ReentrantLock lock = new ReentrantLock(); 
public void untimed() { 
boolean captured = lock. tryLock() ; 
try { 
System.out.printin("tryLock(): " + captured); 
} finally { 
if (captured) 
lock .untock(); 
} 


} 
public void timed() { 
boolean captured = false; 
try { 
Captured = lock. tryLock(2, TimeUnit. SECONDS) ; 
} catch(Interruptedexception e) { 
throw new RuntimeExcept fon(e); 
} 
try { 
System.out.println("tryLock(2, TimeUnit.SECONDS): * ++ 
captured); 
} finally { 
if (captured) 
lock unlock () > 
} 


} 
public static void main(String(] args) { 
final AttemptLocking al = new AttemptLocking(); 
al.untimed(); // True -- lock is available 
al.timed(); // True -- lock is available 
// Now create a separate task to grab the lock: 
new Thread() { k 
{ setDaemon(true); } 
public void run() { 
al.lock. lock(); 
System.out.printin("acquired”); 
} 
}.start(); 
Thread.yield(); // Give the 2nd task a chance 
al.untimed(); // False -- lock grabbed by task 
al.timed();  // False -- lock grabbed by task 
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$ 
} /* Output: 
“tryLock(): true 
tryLock(2, TimeUnit. SECONDS) : true 
acquired 
trylock(): false 
trylock(2, TimeUnit. SECONDS) ; tarse 
Sit: 


ReentrantLock 人 允许 你 尝试 着 获取 但 最 终 未 获取 锁 ， 这 样 如 果 其 他 人 已 经 获取 了 这 个 锁 ， 那 
你 就 可 以 决定 离开 去 执行 其 他 一 些 事情 ， 而 不 是 等 待 直至 这 个 锁 被 释放 ， 就 像 在 untimed0 方 法 
中 所 看 到 的 。 在 timed0 中 ， 做 出 了 尝试 去 获取 锁 ， 该 尝试 可 以 在 2 秒 之 后 失败 (注意 ， 使 用 了 
Java SE5 的 TimeUnit 类 来 指定 时 间 单位 )。 在 main0 中 ， 作为 攻关 而 创建 了 一 个 单独 的 Thread 
它 将 获取 锁 ， 这 使 得 untimed0 和 timed0 方 法 对 某 些 事物 将 产生 竞争 。 

显 式 的 Lock 对 象 在 加 锁 和 释放 锁 方面 ， 相对 于 内 寻 的 synehrorized 锦 来 说 ,还 赋 于 了 你 
细 粒 度 的 控制 力 。 这 对 于 实现 专 有 同步 结构 是 很 有 用 的 ， 例 如 用 于 遍历 链接 列表 中 的 节点 的 节 
节 传 递 的 加 锁 机 制 ( 也 称 为 镇 万 合 )， 这 种 遍历 代码 必须 在 释放 当前 节点 的 锁 之 前 捕获 下 一 个 节 
点 的 锁 。 

21.3.3 原子 性 与 易 变性 

在 有 关 Java 线 程 的 讨论 中 ， 一 个 常 不 正确 的 知识 是 “原子 操作 不 需要 进行 同步 控制 "。 原 子 
操作 是 不 能 被 线程 调度 机 制 中 断 的 操作 ， 一 旦 操作 开始 ， 那 么 它 一 定 可 以 在 可 能 发 生 的 “上 下 
文 切 换 ” 之 前 (切换 到 其 他 线程 执行 ) 执行 完毕 。 依 赖 于 原子 性 是 很 棘手 且 很 危险 的 ， 如 果 你 
是 一 个 并 发 专家 ， 或 者 你 得 到 了 来 自 这 样 的 专家 的 帮助 ， 你 才 应 该 使 用 原子 性 来 代替 同步 。 如 
果 你 认为 自己 足够 聪明 可 以 应 付 这 种 玩 火 似 的 情况 ， 那 么 请 接受 下 面 的 测试 : 

(Goetz 测试 9: 如 果 你 可 以 编写 用 于 现代 微 处 理 器 的 高 性 能 JVM， 那 么 就 有 资格 去 考虑 是 否 
可 以 避免 同步 9 。 

了 解 原子 性 是 很 有 用 的 ， 并 且 要 知道 原子 性 与 其 他 高 级 技术 一 道 ， 在 java.util.coneurrent 类 
库 中 已 经 实现 了 类 此 更 加 巧妙 的 构件 。 但 是 要 坚决 抵挡 住 完全 依 闲 自 己 的 能 力 去 进行 处 理 的 这 
种 欲望 ， 请 看 看 之 前 表述 的 Brian 的 同步 规则 。 

原子 性 可 以 应 用 于 除 Iong 和 double 之 外 的 所 有 基本 类 型 之 上 的 “简单 操作 ”。 对 于 读 取 和 写 
入 除 long 和 double 之 外 的 基本 类 型 变量 这 样 的 操作 ， 可 以 保证 它们 会 被 当 作 不 可 分 (原子 ) 的 操 
作 来 操作 内 存 。 但 是 JVM 可 以 将 64 位 〈iong 和 double 变 量 ) 的 读 取 和 写 信 当 作 两 个 分 离 的 32 位 操 
作 来 执行 ， 这 就 产生 了 在 一 个 读 取 和 写 人 操作 中 间 发 生 上 下 文 切换 ， 从 而 导致 不 同 的 任务 可 以 
看 到 不 正确 结果 的 可 能 性 这 有 时 被 称 为 字 撕 裂 ， 因 为 你 可 能 会 看 到 部 分 被 修改 过 的 数值 )。 但 
是 ， 当 你 定义 long 或 double 变 量 时 ， 如 果 使 用 volatile 关 键 字 ， 就 会 获得 (简单 的 赋值 与 返回 操 
作 的 ) 原子 性 (注意 ， 在 Java SE5 之 前 ，volatile 一 直 未 能 正确 地 工作 )。 不 同 的 VM 可 以 任意 地 
提供 更 强 的 保证 ， 但 是 你 不 应 该 依赖 于 平台 相关 的 特性 。 

因此 ， 原 子 操作 可 由 线程 机 制 来 保证 其 不 可 中 断 ， 专 家 级 的 程序 员 可 以 利用 这 一 点 来 编写 
无 锁 的 代码 ， 这 些 代码 不 需要 被 同步 。 但 是 即便 是 这 样 ， 它 也 是 一 种 过 于 简化 的 机 制 。 有 了 时， 


甚至 看 起 来 应 该 是 安全 的 原子 操作 ， 实 际 上 也 可 能 不 安全 。 本 书 的 读者 通常 不 能 通过 前 面 提 及 





日 以 前 提 到 的 Brian Goetz 命 名 的 测试 。Brian Goetz 是 一 位 并 发 专家 ， 他 对 本 章 有 所 贡献 ， 这 都 源 自 他 那些 半 开 玩 
© 这 个 测试 的 一 个 推论 是 :“ 如 果 某 人 表示 线程 机 制 很 容易 并 且 很 简单 ， 那 么 请 确保 这 个 人 没有 对 你 的 项 目 做 出 
重要 的 决策 。 如 果 这 个 人 已 经 在 这 么 做 了 ， 那 么 你 就 已 经 聊 人 麻烦 之 中 了 。” 
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的 Goetz 测 试 ， 因 此 也 就 不 具备 用 原子 操作 来 替换 同步 的 能 力 。 尝 试 着 移 除 同步 通常 是 一 种 表示 
不 成 熟 优 化 的 信号 ， 并 且 将 会 给 你 招致 大 量 的 麻烦 ， 而 你 却 可 能 没有 收获 多 少 好 处 ， 甚 至 压根 
没有 任何 好 处 。 

在 多 处 理 器 系统 〈 现 在 以 多 核 处 理 器 的 形式 出 现 ， 即 在 单个 芯片 上 有 多 个 CPU) 上 ， 相 对 
于 单 处 理 器 系统 而 言 ， 可 视 性 问题 远 比 原子 性 问题 多 得 多 。 一 个 任务 做 出 的 修改 ， 即 使 在 不 中 
断 的 意义 上 讲 是 原子 性 的 ， 对 其 他 任务 也 可 能 是 不 可 视 的 〈 例 如 ， 修 改 只 是 暂时 性 地 存储 在 本 
地 处 理 器 的 缓存 中 ) ， 因 此 不 同 的 任务 对 应 用 的 状态 有 不 同 的 视图 。 另 一 方面 ， 同 步 机 制 强制 在 
处 理 器 系统 中 ， 一 个 任务 做 出 的 修改 必须 在 应 用 中 是 可 视 的 。 如 果 没 有 同步 机 制 ， 那 么 修改 时 
可 视 将 无 法 确定 。 

volatile 关 键 字 还 确保 了 应 用 中 的 可 视 性 。 如 果 你 将 一 个 域 声明 为 volatile 的 ， 那 么 只 要 对 这 
个 域 产生 了 写 操作 ， 那 么 所 有 的 读 操作 就 都 可 以 看 到 这 个 修改 。 即 便 使 用 了 本 地 缓存 ， 情 况 也 
确实 如 此 ，volatile 域 会 立即 被 写 人 到 主 存 中 ， 而 读 取 操作 就 发 生 在 主 存 中 。 

理解 原子 性 和 易 变 性 是 不 同 的 概念 这 一 点 很 重要 。 在 非 volatile 域 上 的 原子 操作 不 必 刷 新 到 
主 存 中 去 ， 因 此 其 他 读 取 该 域 的 任务 也 不 必 看 到 这 个 新 值 。 如 果 多 个 任务 在 同时 访问 某 个 域 ， 
那么 这 个 域 就 应 该 是 volatile 的 ， 否 则 ， 这 个 域 就 应 该 只 能 经 由 同步 来 访问 。 同 步 也 会 导致 向 主 
存 中 刷新 ， 因 此 如 果 一 个 域 完全 由 synchronized 方 法 或 语句 块 来 防护 ， 那 就 不 必 将 其 设置 为 是 
volatile 的 。 

一 个 任务 所 作 的 任何 写 人 操作 对 这 个 任务 来 说 都 是 可 视 的 ， 因 此 如 果 它 只 需要 在 这 个 任务 
内 部 可 视 ， 那 么 你 就 不 需要 将 其 设置 为 volatile 的 。 

当 一 个 域 的 值 依赖 于 它 之 前 的 值 时 (例如 递增 一 个 计数 器 )，volatile 就 无 法 工作 了 。 如 果 某 
个 域 的 值 受到 其 他 域 的 值 的 限制 ， 那 么 volatile 也 无 法 工作 ， 例 如 Range 类 的 lower 和 upper 边 界 
就 必须 遵循 lower<=upper 的 限制 。 

使 用 volatile 而 不 是 synchronized 的 唯一 安全 的 情况 是 类 中 只 有 一 个 可 变 的 域 。 再 次 提醒 ， 你 
的 第 一 选择 应 该 是 使 用 synchronized 关 键 字 ， 这 是 最 安全 的 方式 ， 而 尝试 其 他 任何 方式 都 是 有 风 
险 的 。 

什么 才 属于 原子 操作 呢 ? 对 域 中 的 值 做 赋值 和 返回 操作 通常 都 是 原子 性 的 ， 但 是 ， 在 C++ 
中 ， 甚至 下 面 的 操作 都 可 能 是 原子 性 的 : 

itt; // Might be atomic in C++ 

i += 2;.// Might be atomic in C++ 
但 是 在 C++ 中 ， 这 要 取决 于 编译 器 和 处 理 器 。 你 无 法 编写 出 依赖 于 原子 性 的 C++ 跨 平台 代码 ， 因 
为 C++ 没有 像 Java (在 Java SESH) 那样 一 致 的 内 存 模型 。 

在 Java 中 ， 上 面 的 操作 肯定 不 是 原子 性 的 ， 正 如 从 下 面 的 方法 所 产生 的 JVM 指 令 中 可 以 看 
到 的 那样 : 

/11:. concurrency/Atomicity.java A 


/1 {Exec: javap -c Atomicity} 


public class Atomicity { 
int i; 
void FIO { i++; } 
void f20 { i += 3; } 
} /* Output: (Sample) 


O 这 在 即将 产生 的 C++ 标准 中 得 到 了 补救 。 
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void f1().; 
Code: 
: aload_@ 
dup 
getfield #2; //Field i:1 
5: iconst_1 
6: iadd 
Fi putfield #2; //Field i:1 
10: return 
void £2(); 
Code: 
9: aload_9 
iz dup 
2: getfield #2; //Field 1:1 
5: ‘iconst_3 
6: iadd 
7: putfield #2; //Field i:I 
10: return 
Wm 


每 条 指令 都 会 产生 一 个 get 和 put， 它 们 之 间 还 有 一 些 其 他 的 指令 。 因 此 在 获取 和 放置 之 间 ， 另 一 
个 任务 可 能 会 修改 这 个 域 ， 所 以 ， 这 些 操作 不 是 原子 性 的 : 


如 果 你 盲目 地 应 用 原子 性 概念 ， 那 么 就 会 看 到 在 下 面 程序 中 的 getValue0 符 合 上 面 的 描述 : 


//: concurrency/AtomicityTest. java 
import java.util.concurrent.*; 


public class AtomicityTest implements Runnable { 
private int i = 9; 
public int getValue() { return i; ) 
private synchronized void evenIncrement() { 1++; i++; } 
+ public void run() { 
while(true) 
evenIncrement(); 


$ 
public static void main(String[] args) { 
ExecutorService exec = Executors.newCachedThreadPool(); 
AtomicityTest at = new AtomicityTest(); 
exec.execute(at); 
while(true) { 
int val = at.getValue(); 
if(val % 2 t= 6) { 
System. out.printin(val); 
System. exit(@); 
} 
} 


} 
} /* Output: (Sample) 
191583767 

Wha 


但 是 ， 该 程序 将 找到 奇数 值 并 终止 。 尽 管 return i 确 实 是 原子 性 操作 ， 但 是 缺少 同步 使 得 其 


数值 可 以 在 处 于 不 稳定 的 中 间 状 态 时 被 读 取 。 除 此 之 外 ， 由 于 i 也 不 是 volatile 的 ， 因 此 还 存在 可 
视 性 问题 。getValue0 和 evenInerement0 必 须 是 synchronized 的 。 在 诸如 此 类 情况 下 ， 只 有 并 发 
专家 才 有 能 力 进行 优化 ， 而 你 还 是 应 该 运用 Brian 的 同步 规则 。 


正如 第 二 个 示例 ， 考 虑 一 些 更 简单 的 事情 : 一 个 产生 序列 数字 的 类 9 。 每 当 nextSerial- 


Number0 被 调用 时 ， 它 必须 向 调用 者 返回 唯一 的 值 : 


//: concurrency/SerialNumberGenerator . java 


© Joshua Blochf (Effective Java Programming Language Guide) (Addison-Wesley, 2001, 19070) 的 启发 ， 本 
书 中 文 版 已 由 机 械 工业 出 版 社 出 版 。 一 一 编辑 注 
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public class SerialNumberGenerator { 
private static volatile int serialNumber = 0; 
public static int nextSerialNumber() { 
return serialNumber++; // Not thread-safe 
} hie 
SerialNumberGenerator 与 你 想象 的 一 样 简单 ， 如 果 你 有 C++ 或 其 他 低层 语言 的 背景 ， 那 么 
可 能 会 期 望 递增 是 原子 性 操作 ， 因 为 C++ 递增 通常 可 以 作为 一 条 微 处 理 器 指令 来 实现 (尽管 不 
是 以 任何 可 靠 的 、 跨 平 台 的 形式 实现 )。 然 而 正如 前 面 注意 到 的 ，Java 递 增 操作 不 是 原子 性 的 ， 
并 且 涉 及 一 个 读 操作 和 一 个 写 操作 ， 所 以 即便 是 在 这 么 简单 的 操作 中 ， 也 为 产生 线程 问题 留 下 
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了 空间 。 正 如 你 所 看 到 的 ， 易 变性 在 这 里 实际 上 不 是 什么 问题 ， 真 正 的 问题 在 于 nextSerial- . 


Number0 在 没有 同步 的 情况 下 对 共享 可 变 值 进行 了 访问 。 

基本 上 ， 如 果 一 个 域 可 能 会 被 多 个 任务 同时 访问 ， 或 者 这 些 任务 中 至 少 有 一 个 是 写 人 任务 ， 
那么 你 就 应 该 将 这 个 域 设置 为 volatile 的 。 如 果 你 将 一 个 域 定义 为 volatile， 那 么 它 就 会 告诉 编译 
器 不 要 执行 任何 移 除 读 取 和 写 人 操作 的 优化 ， 这 些 操作 的 目的 是 用 线程 中 的 局 部 变量 维护 对 这 
个 域 的 精确 同步 。 实 际 上 ， 读 取 和 写 人 都 是 直接 针对 内 存 的 ， 而 却 没有 被 缓存 。 但 是 ，volatile 
并 不 能 对 递增 不 是 原子 性 操作 这 一 事实 产生 影响 。 

为 了 测试 SerialNumberGenerator， 我 们 需要 不 会 耗 尽 内 存 的 集 (Set)， 以 防 需要 花费 很 长 的 
时 间 来 探测 问题 。 这 里 所 示 的 CireularSet 重 用 了 存储 int 数 值 的 内 存 ， 并 假设 在 你 生成 序列 数 时 ， 
产生 数值 覆盖 冲突 的 可 能 性 极 小 。add0 和 contains0 方 法 都 是 synchronized， 以 防止 线程 冲突 : 


//: concurrency/SerialNumberChecker . java 
// Operations that may seem safe are not, 
// when threads are present. 

71 {Args: 4} 

import java.util.concurrent.*; 


// Reuses storage so we don't run out of memory: 
class Circularset { 
private int[] array; 
private int len; 
private int index = 07 
public CircularSet(int size) { 
array = new int(size); 
len = size: 
// Initialize to a value not produced 
// by the SerialNumberGenerator: 
for(int i = 0; 1 < size; i++) 
array(i] = -1; 


public synchronized void add(int i) { 
array[index] = i; 
// Wrap index and write over old elements: 
index = ++index % len; 

} 

public synchronized boolean contains(int val) { 
for(int i = 0; 1 < len; i++) 

if(array(i} == val) return true; 

return false; 

} 

} 


public class SerialNumberChecker { 
private static final int SIZE = 10; 
private static CircularSet serials = 
new CircularSet (1000) ; 
private static ExecutorService exec = 
Executors .newCachedThreadPool () ; 
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static class SerialChecker implements Runnable { 
public void run() { 
while(true) { 
int serial = 
SerialNumberGenerator .nextSerialNumber () ; 
if(serials.contains(serial)) { 
System.out.println("Duplicate: " + serial); 
System.exit(0); 


} 
serials.add(serial); 
} 
3 
} 
public static void main(String[] args) throws Exception { 
for(int i = 0; 1 < SIZE; i++) 
exec. execute(new SerialChecker()); 
// Stop after n seconds if there's an argument: 
if(args.length > 6) { 
TimeUnit. SECONDS: sleep(new Integer (args{@})); 
System.out.printin("No duplicates detected"); 
System.exit(®); 


} r Output: (Sample) 

Duplicate: 8468656 

Whi~ 

SerialNumberChecker 包 含 一 个 静态 的 CircularSet， 它 持 有 所 产生 的 所 有 序列 数 ， 另 外 还 包 
含 一 个 内 伐 的 SerialChecker 类 ， 它 可 以 确保 序列 数 是 唯一 的 。 通 过 创建 多 个 任务 来 竞争 序列 数 ， 
你 将 发 现 这 些 任务 最 终 会 得 到 重复 的 序列 数 ， 如 果 你 运行 的 时 间 足 够 长 的 话 。 为 了 解决 这 个 问 
题 ， 在 nextSerialNumber0 前 面 添加 了 synchronized 关 键 字 。 

对 基本 类 型 的 读 取 和 赋值 操作 被 认为 是 安全 的 原子 性 操作 。 但 是 ， 正 如 你 在 
AtomicityTest,java 中 看 到 的 ， 当 对 象 处 于 不 稳定 状态 时 ， 仍 旧 很 有 可 能 使 用 原子 性 操作 来 访问 
它们 。 对 这 个 问题 做 出 假设 是 坏 手 而 危险 的 ， 最 明智 的 做 法 就 是 遵循 Brian 的 同步 规则 。 

练习 12: (3) 使 用 synchronized 来 修复 Atomicityjava， 你 能 证 明 它 现在 是 安全 的 吗 ? 

练习 13，(1) 使 用 synchronized 来 修复 SerialNumberCheckerjava， 你 能 证 明 它 现在 是 安全 
的 吗 ? 

21.3.4 原子 类 

Java SE5 引 入 了 诸如 AtomicInteger、AtomicLong、AtomicReference 等 特殊 的 原子 性 变量 类 ， 
它们 提供 下 面 形式 的 原子 性 条 件 更 新 操作 : 

boolean compareAndSet(expectedValue, updateValue) : 

这 些 类 被 调整 为 可 以 使 用 在 某 些 现代 处 理 器 上 的 可 获得 的 ， 并 且 是 在 机 器 级 别 上 的 原子 性 ， 
因此 在 使 用 它们 时 ， 通 常 不 需要 担心 。 对 于 常规 编程 来 说 ， 它 们 很 少 会 派 上 用 场 ， 但 是 在 涉及 性 
能 调 优 时 ， 它 们 就 大 有 用 武之 地 了 。 例 如 ， 我 们 可 以 使 用 AtomicInteger 来 重 写 AtomicityTestjava; 


//: concurrency/AtomicIntegerTest. java 
import java.util.concurrent.*; 

import java.util.concurrent.atomic.*; 
import java.util.*; 


public class AtomicIntegerTest implements Runnable { 
private AtomicInteger i = new AtomicInteger (0): 
public int getValue() { return 1.get(); } 
private void evenIncrement() { i.addAndGet(2); } 
public void run() { 
while(true) 
evenIncrement() ; 
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} 
public static void main(String{] args) { 
new Timer().schedule(new TimerTask() { 
public void run() { 
system. err.printIn("Aborting") ; 
System.exit(@); 


} 
}, 5000); // Terminate after 5 seconds 
ExecutorService exec = Executors.newCachedThreadPool(); 
AtomicIntegerTest ait = new AtomicIntegerTest(): 
exec. execute (ait); 
while(true) { 
int val = ait.getValue(); 
if(val % 2 != 6) { 
System. out.printin(val); 
System .exit(9) 1 
} 
} 


} 

} :~ 

这 里 我 们 通过 使 用 AtomicInteger 而 消除 了 synchronized 关 键 字 。 因 为 这 个 程序 不 会 失败 ， 
所 以 添加 了 一 个 Timer， 以 便 在 5 秒 钟 之 后 自动 地 终止 。 

下 面 是 用 AtomicInteger 重 写 的 MutexEvenGenerator.java: 


/1: concurrency/AtomicEvenGenerator .java 

// Atomic classes are occasionally useful in regular code. 
/1 (RunByHand} 

import java.util.concurrent.atomic.*; 





public class AtomicEvenGenerator extends IntGenerator { 
private AtomicInteger currentEvenValue = 
new AtomicInteger (8); 
public int next() { 
return currentevenValue. addAndGet (2); 


} 
public static void main(String[] args) { 
EvenChecker. test (new AtomicEvenGenerator()): 


} 
} M~ 


所 有 其 他 形式 的 同步 再 次 通过 使 用 AtomicInteger 得 到 了 根除 。 

应 该 强调 的 是 ，Atomic 类 被 设计 用 来 构建 java.util.concurrent 中 的 类 ， 因 此 只 有 在 特殊 情况 
下 才 在 自己 的 代码 中 使 用 它们 ， 即 便 使 用 了 也 需要 确保 不 存在 其 他 可 能 出 现 的 问题 。 通 常 依赖 
于 锁 要 更 安全 一 些 (要 么 是 synchronized 关 键 字 ， 要 么 是 显 式 的 Lock 对 象 )。 

练习 14: (4) 创建 一 个 程序 ， 它 可 以 生成 许多 Timer 对 象 ， 这 些 对 象 在 定时 时 间 到 达 后 将 执 
行 某 个 简单 的 任务 。 用 这 个 程序 来 证 明 java.util.Timer 可 以 扩展 到 很 大 的 数目 。 
21.3.5 临界 区 

有 时 ， 你 只 是 希望 防止 多 个 线程 同时 访问 方法 内 部 的 部 分 代码 而 不 是 防止 访问 整个 方法 。 
通过 这 种 方式 分 离 出 来 的 代码 段 被 称 为 临界 区 《critical section) ， 它 也 使 用 synchronized 关 键 字 
建立 。 这 里 ，synchronized 被 用 来 指定 某 个 对 象 ， 此 对 象 的 锁 被 用 来 对 花 括 号 内 的 代码 进行 同步 
控制 : 


synchronized(syncObject) { 
// This code can be accessed 
// by only one task at a time 
} 


这 也 被 称 为 同步 控制 块 ， 在 进入 此 段 代 码 前 ， 必 须 得 到 syncObject 对 象 的 锁 。 如 果 其 他 线程 
已 经 得 到 这 个 锁 ， 那 么 就 得 等 到 镇 被 释放 以 后 ， 才 能 进入 临界 区 。 
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通过 使 用 同步 控制 块 ， 而 不 是 对 整个 方法 进行 同步 控制 ， 可 以 使 多 个 任务 访问 对 象 的 时 间 
性 能 得 到 显著 提高 ， 下 面 的 例子 比较 了 这 两 种 同步 控制 方法 。 此 外 ， 它 也 演示 了 如 何 把 一 个 非 
保护 类 型 的 类 ， 在 其 他 类 的 保护 和 控制 之 下 ， 应 用 于 多 线程 的 环境 ; 


//: concurrency/CriticalSection. java 

// Synchronizing blocks instead of entire methods. Also 
// demonstrates protection of a non-thread-safe class 
// with a thread-safe one. 

package concurrency; 

import java.util.concurrent.*; 
import java.util.concurrent.atomic 
import java.util.*; 





class Pair { // Not thread-safe 
private int x, y; 
public Pair(int x, int y) { 
this.x = x; 
this.y = y; 
} 
public Pair() { this(®, 6); } 
public int getX() { return x; } 
public int getY() { return y; } 
public void incrementx() { x++; } 
public void incrementY() { y++; } 
public String toString() { 
return “x: “+x +", y: 
$ 


public class PairValuesNotEqualException 
extends RuntimeException { 
public PairValuesNotEqualException() { 
super ("Pair values not equal: " + Pair. this); 
$ 
} 
// Arbitrary invariant -- both variables must be equal: 
public void checkState() { 
if(x t= y) 
throw new PairValuesNotEqualException(): 





ty: 


} 


// Protect a Pair inside a thread-safe class: 
abstract class PairManager { 
AtomicInteger checkCounter = new AtomicInteger(®) ; 
protected Pair p = new Pair(): 
private List<Pair> storage = 
Collections. synchronizedt ist (new ArrayList<Pair>()); 
public synchronized Pair getPair() { 
7/ Make a copy to keep the original safe: 
return new Pair(p.getX(), p.getY()); 
} 
// Assume this is a time consuming operation 
protected void store(Pair p) { 
storage.add(p); 
try { 
TimeUnit MILLISECONDS . steep(5®) ; 
} catch(InterruptedException ignore) {} 
} 
public abstract void increment(); 





} 


// Synchronize the entire method: 

class PairManagerl extends PairManager { 

public synchronized void increment() { 
p.incrementX() ; 
p.incrementY(); 
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store(getPair()): 
F 
} 


/1 Use a critical section: 
class PairManager2 extends PairManager { 
public void increment() { 
Pair temp; 
synchrontzed(this) { 
p. incrementX(); 
p. incrementY(); 
temp = getPair(); 
} 
store(temp) ; 
} 
} 


class PairManipulator implements Runnable { 
private PairManager pm; FE 
public PairManipulator(PairManager pm) { 
this.pm = pm; 
} 
Public void run() { 
while(true) 
pm. increment () ; 














} 
public String toString() { 
return “Pair: " + pm.getPair() + 
" checkCounter = ”+ pm.checkCounter.get(); 
2 


} 


class PairChecker implements Runnable { 
private PairManager pm; 
public PairChecker(PairManager pm) { 
this.pm = pm: 
} 
public void run() { 
while(true) { 
pm.checkCounter . incrementAndGet () ; 
pm.getPair().checkState(); 
} 
} 
t 


public class CriticalSection { 
// Test the two different approaches: 
static void 
testApproaches(PairManager pmanl, PairManager pman2) { 
ExecutorService exec = Executors.newCachedThreadPool () ; 
PairManipulator 
pml = new Pairtanipulator(pmani) , 
pm2 = new PairManipulator (pman2) : 
PairChecker 
pcheck1 = new PairChecker(pmani), 
pcheck2 = new PairChecker(pman2); 
exec. execute (pm1) ; 
exec. execute(pm2) ; 
exec.execute (pcheck1) ; 
exec. execute (pcheck2) ; 
try { 
TimeUnit MILLISECONDS. sleep(5@0) ; 
1 } catch(InterruptedException e) { 
| System.out.printin("Sleep interrupted"); 
} 
i System. out. printIn("ps 
| System.exit(0); 











“+ pml + "\npm2: ”+ pm2); 
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} 
public static void main(String[] args) { 
PairManager 
pmanl = new PairManagerl(), 
pman2 = new PairManager2(); 
testApproaches(pman1, pman2); 


¥ 
} /* Output: (Sample) 
pml: Pair: x: 15, y: 15 checkCounter = 272565 
pm2: Pair: x: 16, y: 16 checkCounter = 3956974 
Whim 


正如 注释 中 注 明 的 ，Pair 不 是 线程 安全 的 ， 因 为 它 的 约束 条 件 (虽然 是 任意 的 ) 需要 两 个 
变量 要 维护 成 相同 的 值 。 此 外 ， 如 本 章 前 面 所 述 ， 自 增加 操作 不 是 线程 安全 的 ， 并 且 因 为 没有 
任何 方法 被 标记 为 synchronized ， 所 以 不 能 保证 一 个 Pair 对 象 在 多 线程 程序 中 不 会 被 破坏 。 

你 可 以 想象 一 下 这 种 情况 : 某 人 交 给 你 一 个 非 线程 安全 的 Pair 类 ， 而 你 需要 在 一 个 线程 环 
境 中 使 用 它 。 通 过 创建 PairManager 类 就 可 以 实现 这 一 点 ，PairManager 类 持 有 一 个 Pair 对 象 并 
控制 对 它 的 一 切 访问 。 注 意 唯一 的 public 方 法 是 getPair0 ， 它 是 synchronized 的 。 对 于 抽象 方法 
increment0， 对 increment0 的 同步 控制 将 在 实现 的 时 候 进 行 处 理 。 

至 于 PairManager 类 的 结构 ， 它 的 一 些 功能 在 基 类 中 实现 ， 并 且 其 一 个 或 多 个 抽象 方法 在 派 
生 类 中 定义 ， 这 种 结构 在 设计 模式 中 称 为 模板 方法 9 。 设 计 模式 使 你 得 以 把 变化 封装 在 代码 
里 ， 在 此 ， 发 生变 化 的 部 分 是 模板 方法 incrementO。 在 PairManagerl 中 ， 整 个 increment( 方 法 
是 被 同步 控制 的 ， 但 在 PairManager2 中 ，increment0 方 法 使 用 同步 控制 块 进行 同步 。 注 意 ， 
synchronized 关 键 字 不 属于 方法 特征 签名 的 组 成 部 分 ， 所 以 可 以 在 覆盖 方法 的 时 候 加 上 去 。 

store0 方 法 将 一 个 Pair 对 象 添加 到 了 synchronized ArrayList ,所 以 这 个 操作 是 线程 安全 的 。 
因此 ， 读 方法 不 必 进 行 防护 ， 可 以 放置 在 PairManager2 的 synchronized 语 句 块 的 外 部 。 

PairManipulator 被 创建 用 来 测试 两 种 不 同类 型 的 PairManager， 其 方法 是 在 某 个 任务 中 调 
Hincrement(), ， 而 PairChecker 则 在 另 一 个 任务 中 执行 。 为 了 跟踪 可 以 运行 测试 的 频 度 ， 
PairChecker 在 每 次 成 功 时 都 递增 checkCounter。 在 main0 中 创建 了 两 个 PairManipulator 对 象 ， 
并 允许 它们 运行 一 段 时 间 ， 之 后 每 个 PairManipulator 的 结果 会 得 到 展示 。 

尽管 每 次 运行 的 结果 可 能 会 非常 不 同 ， 但 一 般 来 说 ， 对 于 PairChecker 的 检查 频率 ， 
PairManagerl.increment0 不 允许 有 PairManager2.incerement() 那 样 多 。 后 者 采用 同步 控制 块 进 
行 同步 ， 所 以 对 象 不 加 锁 的 时 间 更 长 。 这 也 是 宁愿 使 用 同步 控制 块 而 不 是 对 整个 方法 进行 同步 
控制 的 典型 原因 ; 使 得 其 他 线程 能 更 多 地 访问 〈 在 安全 的 情况 下 尽 可 能 多 ) 。 

你 还 可 以 使 用 显 式 的 Lock 对 象 来 创建 临界 区 : 

//: concurrency/ExplicitCriticalSection. java 

// Using explicit Lock objects to create critical sections. 


package concurrency; 
import java.util.concurrent.locks.*; 


// Synchronize the entire method: 
class ExplicitPairmanager1 extends PairManager { 
Private Lock lock = new ReentrantLock(); 
public synchronized void increment() { 
tock. lock() ; 
try { 
p. incrementX(); 
p. incrementY() : 
store(getPair()); 


O 参考 《设计 模式 》 (Design Pattermn) ， 作 者 Gamma 等 (Addison-Wesley，1995)。 本 书 英文 版 、 中 文 版 及 双语 版 


均 已 由 机 械 工业 出 版 社 出 版 一 编辑 注 。 
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} finally { 
lock.unlock(): 
2 
fi 
} 


// Use a critical section: 
class ExplicitPairManager2 extends PairManager { 
private Lock lock = new ReentrantLock() ; 
public void increment() { 
Pair temp; 
lock. Lock () ; 
try { 
Pp. incrementx(); 
p. incrementY(); 
temp = getPair(); 
} finally { 
lock.unlock(); 


} 
store(temp); 
$ 


public class ExplicitCriticalSection { 
public static void main(String{] args) throws Exception { 
PairManager 
pmanl = new ExplicitPairManager1(), 
pman2 = new ExplicitPairManager2(); 
CriticalSection.testApproaches(pmani, pman2); 


} 
} /* Output: (Sample) 
pml: Pair: x: 15, y: 15 checkCounter = 174035 
pm2: Pair: x: 16, y: 16 checkCounter = 2608588 
Whim 


这 里 复 用 了 CriticalSeetion.java 的 绝 大 部 分 ， 并 创建 了 新 的 使 用 显 式 的 Lock 对 象 的 
PairManager 类 型 。ExplicitPairManager2 展 示 了 如 何 使 用 Lock 对 象 来 创建 临界 区 ， 而 对 store() 
的 调用 则 在 这 个 临界 区 的 外 部 。 

21.3.6 在 其 他 对 象 上 同步 

Synchronized 块 必须 给 定 一 个 在 其 上 进行 同步 的 对 象 ， 并 且 最 合理 的 方式 是 ， 使 用 其 方法 正 
在 被 调用 的 当前 对 象 ，synchronized(this)， 这 正 是 PairManager2 所 使 用 的 方式 。 在 这 种 方式 中 ， 
如 果 获得 了 synchronized 块 上 的 锁 ， 那 么 该 对 象 其 他 的 synchronized 方 法 和 临界 区 就 不 能 被 调用 

了。 因此， 如 果 在 this 上 同步 ， 临 界 区 的 效果 就 会 直接 缩小 在 同步 的 范围 内 。 
有 时 必须 在 另 一 个 对 象 上 同步 ， 但 是 如 果 你 要 这 么 做 ， 就 必须 确保 所 有 相关 的 任务 都 是 在 
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同一 个 对 象 上 同步 的 。 下 面 的 示例 演示 了 两 个 任务 可 以 同时 进入 同一 个 对 象 ， 只 要 这 个 对 象 上 


的 方法 是 在 不 同 的 锁 上 同步 的 即 可 : 


//: concurrency/SyncObject.java 
// Synchronizing on another object. 
import static net.mindview.util.Print.*; 


class DualSynch { 
private Object syncObject = new Object(); 
public synchronized void fO { 
for(int i = @; i < 5; i++) { 
print("f()"); 
Thread. yield(); 


} 
public void g() { 
synchronized(syncObject) { 





‘(i 
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for(int 1 = @; i < 5; i+) { 
print("g()"); 
Thread.yield(); 
4 
} 
} 
} 


public class SyncObject { 
public static void main(String(] args) { 
final DualSynch ds = new DualSynch(); 
new Thread() { 
public void run() { 
ds.fO; 


} 
}.startO); 
ds.g0; 


} 
} /* Output: (Sample) 
g0 
tO 
BO 
tO 
BO. 
tO 
g0 
tO 
BO 
tO 
Whim 


DualSync.fQ) (通过 同步 整个 方法 ) 在 this 同 步 ， 而 g0 有 一 个 在 syneObjecet 上 同步 的 
synchronized 块 。 因 此 ， 这 两 个 同步 是 互相 独立 的 。 通 过 在 main0 中 创建 调用 f0 的 Thread 对 这 一 
点 进行 了 演示 ， 因 为 main0 线 程 是 被 用 来 调用 g0 的 。 从 输出 中 可 以 看 到 ， 这 两 个 方式 在 同时 运 
行 ， 因此 任何 一 个 方法 都 没有 因为 对 另 一 个 方法 的 同步 而 被 阻塞 。 

练习 15: (1) 创建 一 个 类 ， 它 具有 三 个 方法 ， 这 些 方法 包含 一 个 临界 区 ， 所 有 对 该 临界 区 的 
同步 都 是 在 同一 个 对 象 上 的 。 创 建 多 个 任务 来 演示 这 些 方法 同时 只 能 运行 一 个 。 现 在 修改 这 些 
方法 ， 使 得 每 个 方法 都 在 不 同 的 对 象 上 同步 ， 并 展示 所 有 三 个 方法 可 以 同时 运行 。 

练习 16: (1) 使 用 显 式 的 Lock 对 象 来 修改 练习 15。 

21.3.7 线程 本 地 存储 

防止 任务 在 共享 资源 上 产生 冲突 的 第 二 种 方式 是 根除 对 变量 的 共享 。 线 程 本 地 存储 是 一 种 
自动 化 机 制 ， 可 以 为 使 用 相同 变量 的 每 个 不 同 的 线程 都 创建 不 同 的 存储 。 因 此 ， 如 果 你 有 5 个 线 
程 都 要 使 用 变量 x 所 表示 的 对 象 ， 那 线程 本 地 存储 就 会 生成 5 个 用 于 x 的 不 同 的 存储 块 。 主 要 是 ， 
它们 使 得 你 可 以 将 状态 与 线程 关联 起 来 。 

创建 和 管理 线程 本 地 存储 可 以 由 java4ang.ThreadLocal 类 来 实现 ， 如 下 所 示 : 


//: concurrency/ThreadLocalVariableHolder .java 

1/ Automatically giving each thread its own storage. 
import java.util.concurrent.*; 

import java.util.*: 


class Accessor implements Runnable { 

private final int id; 

public Accessor(int idn) { id = idn; } 

public void run() { 

while(!Thread.currentThread().isInterrupted()) { 

ThreadLocalVariableHolder. increment (); 
System. out.printin(this) ; 
Thread.yield(); 
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} 
} 
public String tostring() { 
return "#" + id +": * 
ThreadLocalVariableHiolder. get(): 
} 
} 


public class ThreadLocalVariableHolder { 
private static ThreadLlocal<Integer> value = 
new ThreadLocal<Integer>() { 
private Random rand = new Random(47); 
protected synchronized Integer initialValue() { 
return rand.nextInt(19008) ; 
} 
}; 
public static void increment() { 
value.set(value.get() + 1); 
} 
public static int get() { return value.get(); } 
public static void main(String[] args) throws Exception { 
ExecutorService exec = Executors.newCachedThreadPool(); 
for(int 1 = @; i < 5; i++) 
exec. execute(new Accessor (i)): 
TimeUnit. SECONDS. sleep(3); // Run for a while 
exec. shutdownNow() ; /1/ ALL Accessors will quit 


} 

} /* Output: (Sample) 
#0: 9259 
#1: 556 
#2: 6694 
#3: 1862 
#4: 962 
#0: 9260 
#1: 557 
#2: 6695 
: 1863 
: 963 





*/L1 :~ 

ThreadLocal 对 象 通常 当 作 静 态 域 存储 。 在 创建 ThreadLocal 时 ， 你 只 能 通过 get0 和 set( 方 
法 来 访问 该 对 象 的 内 容 ， 其 中 ，get0 方 法 将 返回 与 其 线程 相关 联 的 对 象 的 副本 ， 而 set0 会 将 参 
数 插入 到 为 其 线程 存储 的 对 象 中 ， 并 返回 存储 中 原 有 的 对 象 。inerement() 和 get0 方 法 在 
ThreadLocalVariableHolder 中 演示 了 这 一 点 。 注 意 ，inerement0 和 get0 方 法 都 不 是 synchronized 
的 ， 因 为 ThreadLocal 保 证 不 会 出 现 竞争 条 件 。 

当 运 行 这 个 程序 时 ， 你 可 以 看 到 每 个 单独 的 线程 都 被 分 配 了 自己 的 存储 ， 因 为 它们 每 个 都 
需要 跟踪 自己 的 计数 值 ， 即 便 只 有 一 个 ThreadLocalVariableHolder 对 象 。 


21.4 终结 任务 


在 前 面 的 某 些 示例 中 , cancel0 和 isCanceled0 方 法 被 放 到 了 一 个 所 有 任务 都 可 以 看 到 的 类 中 。 
这 些 任务 通过 检查 isCanceled0 来 确定 何 时 终止 它们 自己 ， 对 于 这 个 问题 来 说 ， 这 是 一 种 合理 的 
方式 。 但 是 ， 在 某 些 情况 下 ， 任 务必 须 更 加 突然 地 终止 。 本 节 你 将 学 习 到 有 关 这 种 终止 的 各 类 
话题 和 问题 。 

首先 ， 让 我 们 观察 一 个 示例 ， 它 不 仅 演示 了 终止 问题 ， 而 且 还 是 一 个 资源 共享 的 示例 。 
21.4.1 装饰 性 花园 

在 这 个 仿真 程序 中 ， 花 园 委员 会 希望 了 解 每 天 通过 多 个 大 门 进 入 公园 的 总 人 数 。 每 个 大 门 
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都 有 一 个 十 字 转 门 或 某 种 其 他 形式 的 计数 器 ， 并 且 任何 一 个 十 字 转 门 的 计数 值 递增 时 ， 就 表示 
公园 中 的 总 人 数 的 共享 计数 值 也 会 递增 。 


//: concurrency/OrnamentalGarden. java 
import java.util.concurrent.*; 

import java.util. 
import static net.mindview.util.Print.*; 





class Count { 
private int count = 0; 
private Random rand = new Random(47); 
// Remove the synchronized keyword to see counting fail: 
public synchronized int increment() { 











1179} int temp = count; 





if(rand.nextBoolean()) // Yield half the time 
Thread.yield(); 
return (count = ++temp); 
} 
public synchronized int value() { return count; } 
} 


class Entrance implements Runnable { 
private static Count count = new Count(); 
private static List<Entrance> entrances = 
new ArrayList<Entrance>(); 
private int number = 0; 
// Doesn't need synchronization to read: 
private final int id; 
private static volatile boolean canceled = false; 
// Atomic operation on a volatile fiel 
public static void cancel() { canceled 
public Entrance(int id) { 
this.id = id: 
// Keep this task in a list. Also prevents 
// garbage collection of dead tasks: 
entrances. add(this) ; 
} 
public void run() { 
while(!canceled) { 
synchronized(this) { 
++number ; 
} 
print(this + " Total: " + count.increment()); 
try { 
TimeUnit MILLISECONDS. sleep (100) ; 
} catch(InterruptedException e) { 
print(*sleep interrupted"); 
} 
} 
print("Stopping ”+ this); 
} 
public synchronized int getValue() { return number; } 
public String toString() { 
return "Entrance " + id + ": " + getValue(); 
} 
public static int getTotalCount() { 
return count.value(); 
[1180 } 
en public static int sumEntrances() { 
int sum = @; 
for(Entrance entrance : entrances) 
sum += entrance. getValue(); 
return sum; 
} 





true; } 


} 


public class OrnamentalGarden { 
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public static void main(String{) args) throws Exception { 
ExecutorService exec = Executors.newCachedThreadPool () ; 
for(int i = @; 1 < S; i++) 
exec.execute(new Entrance(i)); 
// Run for a while, then stop and collect the data: 
TimeUnit . SECONDS. sleep (3); 
Entrance. cancel () ; 
exec. shutdown() ; 
if (fexec. awaitTermination(25@, TimeUnit.MILLISECONDS)) 
print("Some tasks were not terminated!"); 
print("Total: ”+ Entrance.getTotalCount()); 
print("Sum of Entrances: ”+ Entrance.sumEntrances()); 


} 
} /* Output: (Sample) 


Entrance 0; 1 Total: 1 
Entrance 2: 1 Total: 3 
Entrance 1: 1 Total: 2 
Entrance 4: 1 Total: 5 
Entrance 3: 1 Total: 4 
Entrance 2: 2 Total: 6 
Entrance 4: 2 Total: 7 
Entrance 0: 2 Total: 8 
Entrance 3: 29 Total: 143 
Entrance 0: 29 Total: 144 
Entrance 4: 29 Total: 145 
Entrance 2: 30 Total: 147 
Entrance 1: 30 Total: 146 
Entrance @; 30 Total: 149 


Entrance 3: 30 Total: 148 
Entrance 4: 30 Total: 150 
Stopping Entrance 2: 30 
Stopping Entrance 1: 30 
Stopping Entrance @: 30 
Stopping Entrance 3: 30 
Stopping Entrance 4: 30 
Total: 150 

Sum of Entrances: 150 
Whim 


这 里 使 用 单个 的 Count 对 象 来 跟踪 花园 参观 者 的 主 计数 值 ， 并 且 将 其 当 作 Entrance 类 中 的 一 
个 静态 域 进行 存储 。Count,increment0 和 Count.value0 都 是 synchronized 的 ， 用 来 控制 对 count 
域 的 访问 。inerement0 方 法 使 用 了 Random 对 象 ， 目 的 是 在 从 把 count 读 取 到 temp 中 ， 到 递增 
temp 并 将 其 存储 回 count 的 这 段 时 间 里 ， 有 大 约 一 半 的 时 间 产 生 让 步 。 如 果 你 将 inerementO 上 的 
synchronized 关 键 字 注 释 掉 ， 那 么 这 个 程序 就 会 崩溃 ， 因 为 多 个 任务 将 同时 访问 并 修改 count 
(yield0 会 使 问题 更 快 地 发 生 ) 。 

每 个 Entrance 任 务 都 维护 着 一 个 本 地 值 number， 它 包含 通过 某 个 特定 入口 进 入 的 参观 者 的 
数量 。 这 提供 了 对 count 对 象 的 双重 检查 ， 以 确保 其 记录 的 参观 者 数量 是 正确 的 。Entrance.run0 
只 是 递增 number 和 count 对 象 ， 然 后 休眠 100 毫 秒 。 

因为 Entrance.canceled 是 一 个 volatile 布 尔 标志 ， 而 它 只 会 被 读 取 和 赋值 (不 会 与 其 他 域 组 
合 在 一 起 被 读 取 )， 所 以 不 需要 同步 对 其 的 访问 ， 就 可 以 安全 地 操作 它 。 如 果 你 对 诸如 此 类 的 情 
况 有 任何 疑虑 ， 那 么 最 好 总 是 使 用 synchronized 。 

这 个 程序 在 以 稳定 的 方式 关闭 所 有 事物 方面 还 有 一 些小 麻烦 ， 其 部 分 原因 是 为 了 说 明 在 终 
止 多 线程 程序 时 你 必须 相当 小 心 ， 而 另 一 部 分 原因 是 为 了 演示 interrupt0 的 值 ， 稍 后 你 将 学 习 有 
关 这 个 值 的 知识 。 

在 3 秒 钟 之 后 ，main0 向 Entrance 发 送 static cancel0 消 息 ， 然 后 调用 exec 对 象 的 shutdown0 方 
法 ， 之 后 调用 exec 上 的 awaitTermination0 方 法 。ExecutorService.awaitTermination0 等 待 每 个 任 
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务 结束 ， 如 果 所 有 的 任务 在 超时 时 间 达 到 之 前 全 部 结束 ， 则 返回 true， 否 则 返回 false， 表 示 不 是 
所 有 的 任务 都 已 经 结束 了 。 尽 管 这 会 导致 每 个 任务 都 退出 其 ran0 方 法 ， 并 因此 作为 任务 而 终止， 
但 是 Entrance 对 象 仍旧 是 有 效 的 ， 因 为 在 构造 器 中 ， 每 个 Entrance 对 象 都 存储 在 称 为 entrances 的 
静态 List<Entrance> 中 。 因 此 ，sumEntrances0 仍 旧 可 以 作用 于 这 些 有 效 的 Entrance 对 象 。 

当 这 个 程序 运行 时 ， 你 将 看 到 ， 在 人 们 通过 十 字 转 门 时 ， 将 显示 总 人 数 和 通过 每 个 人 口 的 
人 数 。 如 果 移 除 CountinerementO 上 面 的 synchronized 声 明 ， 你 将 会 注意 到 总 人 数 与 你 的 期 望 有 
差异 ， 每 个 十 字 转 门 统计 的 人 数 将 与 count 中 的 值 不 同 。 只 要 用 互 斥 来 同步 对 Count 的 访问 ， 问 
题 就 可 以 解决 了 。 请 记 住 ，Count.incrementO 通 过 使 用 temp 和 yield0， 增 加 了 失败 的 可 能 性 。 
在 真正 的 线程 问题 中 ， 失 败 的 可 能 性 从 统计 学 角度 看 可 能 非常 小 ， 因 此 你 可 能 很 容易 就 掉 进 了 
轻信 所 有 事物 都 将 正确 工作 的 陷阱 里 。 就 像 在 上 面 的 示例 中 ， 有 些 还 未 发 生 的 问题 就 有 可 能 会 
隐藏 起 来 ， 因 此 在 复审 并 发 代码 时 ， 要 格外 地 仔细 。 

练习 17: (2) 创建 一 个 辐射 计数 器 ， 它 可 以 具有 任意 数量 的 传感器 。 

21.4.2 在 阻塞 时 终结 

前 面 示例 中 的 Entrance.run0 在 其 循环 中 包含 对 sleep0 的 调用 。 我 们 知道 ，sleep0 最 终 将 唤 
醒 ， 而 任务 也 将 返回 循环 的 开始 部 分 ， 去 检查 canceled 标 志 ， 从 而 决定 是 否 跳出 循环 。 但 是 ， 
sleep0 一 种 情况 ， 它 使 任务 从 执行 状态 变 为 被 阻塞 状态 ， 而 有 时 你 必须 终止 被 阻塞 的 任务 。 

线程 状态 

一 个 线程 可 以 处 于 以 下 四 种 状态 之 一 : 

1) 新 建 (new)， 当 线程 被 创建 时 ， 它 只 会 短暂 地 处 于 这 种 状态 。 此 时 它 已 经 分 配 了 必需 的 
系统 资源 ， 并 执行 了 初始 化 。 此 刻 线程 已 经 有 资格 获得 CPU 时 间 了 ， 之 后 调度 器 将 把 这 个 线程 
转变 为 可 运行 状态 或 阻塞 状态 。 

2) 就 绪 (Runnable): 在 这 种 状态 下 ， 只 要 调度 器 把 时 间 片 分 配给 线程 ， 线 程 就 可 以 运行 。 
也 就 是 说 ， 在 任意 时 刻 ， 线 程 可 以 运行 也 可 以 不 运行 。 只 要 调度 器 能 分 配 时 间 片 给 线程 ， 它 就 
可 以 运行 ， 这 不 同 于 死亡 和 阻塞 状态 。 

3) 阻塞 (Blocked); 线程 能 够 运行 ， 但 有 某 个 条 件 阻 止 它 的 运行 。 当 线程 处 于 阻塞 状态 时 ， 
调度 器 将 忽略 线程 ， 不 会 分 配给 线程 任何 CPU 时 间 。 直 到 线程 重新 进入 了 就 绪 状 态 ， 它 才 有 可 
能 执行 操作 。 

4) EE (Dead): 处 于 死亡 或 终止 状态 的 线程 将 不 再 是 可 调度 的 ， 并 且 再 也 不 会 得 到 CPU 时 
间 ， 它 的 任务 已 结束 ， 或 不 再 是 可 运行 的 。 任 务 死亡 的 通常 方式 是 从 run() 方 法 返回 ， 但 是 任务 
的 线程 还 可 以 被 中 断 ， 你 将 要 看 到 这 一 点 。 

进入 阻塞 状态 

一 个 任务 进入 阻塞 状态 ， 可 能 有 如 下 原因 : 

1) 通过 调用 sleep(milliseconds) 使 任务 进入 休眠 状态 ， 在 这 种 情况 下 ， 任 务 在 指定 的 时 间 内 
不 会 运行 。 : 

2) 你 通过 调用 waitO 使 线程 挂 起 。 直 到 线程 得 到 了 notify0 或 notifyANO 消 息 (或 者 在 Java 
SE5 的 java.util.concurrent 类 库 中 等 价 的 signal0 或 signalAli0 消 息 ) ， 线 程 才 会 进入 就 绪 状态 。 我 
们 将 在 稍 后 的 小 节 中 验证 这 一 点 。 

3) 任务 在 等 待 某 个 输入 /输出 完成 。 

4) 任务 试图 在 某 个 对 象 上 调用 其 同步 控制 方法 ， 但 是 对 象 锁 不 可 用 ， 因 为 另 一 个 任务 已 经 
获取 了 这 个 锁 。 

在 较 早 的 代码 中 ， 也 可 能 会 看 到 用 suspend0 和 resume0 来 阻塞 和 唤醒 线程 ， 但 是 在 现代 Java 
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中 这 些 方法 被 废止 了 (因为 可 能 导致 死 锁 )， 所 以 本 书 不 讨论 这 些 内 容 。stop0 方 法 也 已 经 被 废 
止 了 ， 因 为 它 不 释放 线程 获得 的 锁 ， 并 且 如 果 线程 处 于 不 一 致 的 状态 〈 受 损 状 态 ) ， 其 他 任务 可 
以 在 这 种 状态 下 浏览 并 修改 它们 。 这 样 所 产生 的 问题 是 微妙 而 难以 被 发 现 的 。 

现在 我 们 需要 查看 的 问题 是 ， 有 时 你 希望 能 够 终止 处 于 阻塞 状态 的 任务 。 如 果 对 于 处 于 阻 
塞 状态 的 任务 ， 你 不 能 等 待 其 到 达 代码 中 可 以 检查 其 状态 值 的 某 一 点 ， 因 而 决定 让 它 主动 地 终 
止 ， 那 么 你 就 必须 强制 这 个 任务 跳出 阻塞 状态 。 
21.4.3 中 断 

正如 你 所 想象 的 ， 在 Runnable.run0 方 法 的 中 间 打 断 它 ， 与 等 待 该 方法 到 达 对 cancel 标 志 的 
测试 ， 或 者 到 达 程 序 员 准备 好 离开 该 方法 的 其 他 一 些 地 方 相 比 ， 要 棘手 得 多 。 当 你 打 断 被 阻塞 
的 任务 时 ， 可 能 需要 清理 资源 。 正 因为 这 一 点 ， 在 任务 的 run0 方 法 中 间 打 断 ， 更 像 是 抛 出 的 异 
常 ， 因 此 在 Java 线 程 中 的 这 种 类 型 的 异常 中 断 中 用 到 了 异常 。 (这 会 滑 向 异常 的 不 恰当 用 法 , A, 
为 这 意味 着 你 经 常用 它们 来 控制 执行 流程 )。 为 了 在 以 这 种 方式 终止 任务 时 ， 返 回 众所周知 的 良 
好 状态 ， 你 必须 仔细 考虑 代码 的 执行 路 径 ， 并 仔细 编写 catch 子 句 以 正确 清除 所 有 事物 。 

Thread 类 包含 interrupt0 方 法 ， 因 此 你 可 以 终止 被 阻塞 的 任务 ， 这 个 方法 将 设置 线程 的 中 断 
状态 。 如 果 一 个 线程 已 经 被 阻塞 ， 或 者 试图 执行 一 个 阻塞 操作 ， 那 么 设置 这 个 线程 的 中 断 状态 
将 抛 出 InterruptedException。 当 抛 出 该 异常 或 者 该 任务 调用 Thread.interrupted0 时 ， 中 断 状态 
将 被 复位 。 正 如 你 将 看 到 的 ，Thread.interruptedO 提 供 了 离开 run0 循 环 而 不 抛 出 异常 的 第 二 种 
方式 。 

为 了 调用 interrupt0， 你 必须 持 有 Thread 对 象 。 你 可 能 已 经 注意 到 了 ， 新 的 concurrent 类 库 
似乎 在 避免 对 Thread 对 象 的 直接 操作 ， 转 而 尽量 通过 Executor 来 执行 所 有 操作 。 如 果 你 在 
Executor 上 调用 shutdownNow0， 那 么 它 将 发 送 一 个 interruptO 调 用 给 它 启 动 的 所 有 线程 。 这 么 
做 是 有 意义 的 ， 因 为 当 你 完成 工程 中 的 某 个 部 分 或 者 整个 程序 时 ， 通 常会 希望 同时 关闭 某 个 特 
定 Executor 的 所 有 任务 。 然 而 ， 你 有 时 也 会 希望 只 中 断 某 个 单一 任务 。 如 果 使 用 Executor， 那 
么 通过 调用 submit0 而 不 是 executor0 来 启动 任务 ,就 可 以 持 有 该 任务 的 上 下 文 。submit0 将 返回 
一 个 泛 型 Future<?>， 其 中 有 一 个 未 修 饰 的 参数 ， 因 为 你 永远 都 不 会 在 其 上 调用 get0 一 — 持 有 这 
种 Future 的 关键 在 于 你 可 以 在 其 上 调用 cancel0， 并 因此 可 以 使 用 它 来 中 断 某 个 特定 任务 。 如 果 
你 将 true 传 递 给 cancel0， 那 么 它 就 会 拥有 在 该 线程 上 调用 interrupt0 以 停止 这 个 线程 的 权限 。 
此 ，eancel0 是 一 种 中 断 由 Executor 启 动 的 单个 线程 的 方式 。 

下 面 的 示例 用 Executor 展 示 了 基本 的 interrupt0 用 法 : 


//: concurrency/Interrupting.java 

// Interrupting a blocked thread. 
import java.util.concurrent.*; 

‘import java.io.*; 

import static net.mindview.util.Print.*; 


class SleepBlocked implements Runnable { 
public void run() { 
try { 
TimeUnit. SECONDS. sleep(198) ; 
} catch(Interruptedexception e) { 
print("InterruptedException"); 


} 
print("Exiting SleepBlocked.run()"); 


日 但 是 ， 异 常 从 来 都 不 能 异步 地 传递 。 因 此 ， 在 指令 /方法 调用 的 中 间 突 然 中 断 没有 任何 危险 。 只 要 在 使 用 对 象 
互 斥 机 制 ( 与 synehronized 关 键 字 相对 ) 时 使 用 try-finally 惯 用 法 ， 如 果 抛 出 异常 ， 这 些 互 斥 就 会 自动 被 释放 。 
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class IO0Blocked implements Runnable { 
private InputStream in; 
public I0Blocked(InputStream is) { in = is; } 
public void run() { 
try { 
print("Waiting for read(): 
in.read(); 
} catch(IOException e) { 
if (Thread. currentThread().isInterrupted()) { 
print("Interrupted from blocked 1/0"); 
} else { 
throw new RuntimeException(e); 
} 
} 
print("Exiting 10Blocked. run()"); 
} 
} 





class SynchronizedBlocked implements Runnable { 
public synchronized void f() { 
while(true) // Never releases lock 
Thread. yield(); 


} 
public SynchronizedBlocked() { 
new Thread() { 
public void run() { 
f(); // Lock acquired by this thread 
} k 
}estart(); 
} 
public void run() { 
print("Trying to call f0"); 
tO: 
print("Exiting SynchronizedBlocked.run()"); 
1 
} 


public class Interrupting { 
private static ExecutorService exec = 
Executors.newCachedThreadPool () ; 
static void test(Runnable r) throws InterruptedExcept ion{ 
Future<?> f = exec.submit(r); 
TimeUnit MILLISECONDS. sleep (108) ; 
print("Interrupting ”+ r.getClass().getName()); 
f.cancel(true); // Interrupts if running 
print("Interrupt sent to " + r.getClass().getName()); 
} 
public static void main(String{] args) throws Exception { 
test (new SleepBlocked()); 
test (new I0Blocked(System.in)); 
test (new SynchronizedBlocked()); 
TimeUnit . SECONDS. steep(3); 
print("Aborting with System.exit(@)"); 
System .exit(9); // ... since last 2 interrupts failed 
} 
} /* Output: (95% match) 
Interrupting SleepBlocked 
InterruptedException 
Exiting SleepBlocked.run() 
Interrupt sent to SleepBlocked 
Waiting for read(): 
Interrupting IOBlocked 
Interrupt sent to I0Blocked 
Trying to call fO 
Interrupting SynchronizedBlocked 
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Interrupt sent to SynchronizedBlocked 
Aborting with System.exit(8) 
Whim 


上 面 的 每 个 任务 都 表示 了 一 种 不 同类 型 的 阻塞 。SleepBlock 是 可 中 断 的 阻塞 示例 ， 而 
IOBlocked 和 SynchronizedBlocked 是 不 可 中 断 的 阻塞 示例 。。 这 个 程序 证 明 1/O 和 在 synchronized 
块 上 的 等 待 是 不 可 中 断 的 ， 但 是 通过 浏览 代码 ， 你 也 可 以 预见 到 这 一 点 一 一 无 论 是 /O 还 是 尝试 
调用 synchronized 方 法 ， 都 不 需要 任何 InterruptedException 处 理 器 。 

前 两 个 类 很 简单 直观 : 在 第 一 个 类 中 run0 方 法 调用 了 sleep0, 而 在 第 二 个 类 中 调用 了 read0。 
但 是 ， 为 了 演示 SynchronizedBlock ， 我 们 必须 首先 获取 锁 。 这 是 通过 在 构造 器 中 创建 匿名 的 
Thread 类 的 实例 来 实现 的 ， 这 个 匿名 Thread 类 的 对 象 通过 调用 f0 获 取 了 对 象 锁 (这 个 线程 必须 
有 别 于 为 SynchronizedBlock 驱 动 run0 的 线程 ， 因 为 一 个 线程 可 以 多 次 获得 某 个 对 象 锁 ) 。 由 于 
f0 永 远 都 不 返回 ， 因 此 这 个 锁 永 远 不 会 释放 ， 而 SynchronizedBlock.run0 在 试图 调用 f0， 并 阻塞 
以 等 待 这 个 锁 被 释放 。 

从 输出 中 可 以 看 到 ， 你 能 够 中 断 对 sleep0 的 调用 (或 者 任何 要 求 抛 出 InterruptedException 
的 调用 )。 但 是 ， 你 不 能 中 断 正在 试图 获取 synchronized 锁 或 者 试图 执行 WO 操作 的 线程 。 这 有 点 
令 人 烦恼 ， 特 别 是 在 创建 执行 1JO 的 任务 时 ， 因 为 这 意味 着 MO 具 有 锁 住 你 的 多 线程 程序 的 潜在 可 
能 。 特 别 是 对 于 基于 Web 的 程序 ， 这 更 是 关 平 利害 。 

对 于 这 类 问题 ， 有 一 个 略 显 笨拙 但 是 有 时 确实 行 之 有 效 的 解决 方案 ， 即 关闭 任务 在 其 上 发 
生 阻塞 的 底层 资源 : 


/1: concurrency/CloseResource. java 
// Interrupting a blocked task by 
// closing the underlying resource. 
1/ {RunByHand) 
import java.net.*; 
import java.util.concurrent.*; 

import java.io.*; 

import static net.mindview.util.Print.*; 








public class CloseResource { 

public static void main(String(] args) throws Exception { 
ExecutorService exec = Executors.newCachedThreadPool () ; 
ServerSocket server = new ServerSocket (8088) ; 
InputStream socketInput = 

new Socket "localhost", 880) .getInputStream() ; 

exec.execute(new IOBlocked (socket Input)) ; 
exec.execute(new I0Blocked (System. in)); 
TimeUnit MILLISECONDS. sleep(16) ; 
print("Shutting down all threads"): 
exec. shutdownNow() ; 
TimeUnit . SECONDS. sleep(1); 
print("Closing ”+ socketInput.getClass().getName()); 
socketInput.close(); // Releases blocked thread 
TimeUnit. SECONDS. sleep(1); 
print("Closing " + System, in.getClass() .getName()) ; 
System. in.close(); // Releases blocked thread 


} 
} /* Output: (85% match) 
Waiting for read(): 
Waiting for read(): 
Shutting down all threads 
Closing java.net. SocketInputStream 
Interrupted from blocked 1/0 


O 某 些 版 本 的 JDK 还 提供 对 InterruptedIOException 的 支持 。 但 是 ， 这 只 是 部 分 实现 ， 而 且 只 在 某 些 平台 上 可 用 。 


如 果 抛 出 这 个 异常 ， 它 会 导致 IO 对象 不 可 用 。 未 来 的 版 本 不 太 可 能 继续 支持 这 个 异 当 。 
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Exiting 10Blocked.run() 

Closing java.io.BufferedInputStream 
Exiting I0Blocked.run() 

Whim 


在 shutdownNow0 被 调用 之 后 以 及 在 两 个 输入 流 上 调用 close0 之 前 的 延迟 强调 的 是 一 旦 底层 
资源 被 关闭 ， 任 务 将 解除 阻塞 。 请 注意 ， 有 一 点 很 有 趣 ，interrupt0 看 起 来 发 生 在 关闭 Socket 而 
不 是 关闭 System.in 的 时 刻 。 

幸运 的 是 ， 在 第 18 章 中 介绍 的 各 种 nio 类 提供 了 更 人 性 化 的 VO 中 断 。 被 阻塞 的 nio 通 道 会 自 
动 地 响应 中 断 ; 


//: concurrency/NIOInterruption. java 
// Interrupting a blocked NIO channel. 
import java.net.*; 

import java.nio.*; 

import java.nio.channets.*; 

import java.util.concurrent.*; 

import java.io. 
import static net.mindview.util.Print 








class NIOBlocked implements Runnable { 
private final SocketChannel sc; 
public NIOBlocked(SocketChannel sc) { this.sc = sc; } 
public void run() { 
try { 
print("Waiting for read() in * + this); 
sc.read(ByteBuffer.allocate(1)); 
} catch(ClosedByInterruptException e) { 
print ("ClosedByInterruptException") ; 
} catch(AsynchronousCloseException e) { 
print ("AsynchronousCloseException”) ; 
} catch(I0Exception e) { 
throw new RuntimeException(e); 
} 
print("Exiting NIOBlocked.run() ”+ this); 
} 
} 


public class NIOInterruption { 

public static void main(String[] args) throws Exception { 
ExecutorService exec = Executors.newCachedThreadPool(); 
ServerSocket server = new ServerSocket (8080) ; 
InetSocketAddress isa = 

new InetSocketAddress("Localhost”, 8080) ; 

SocketChannel scl = SocketChannel.open(isa); 
SocketChannel sc2 = SocketChannel.open(isa): 
Future<?> f = exec. submit (new NIOBlocked(sc1)); 
exec.execute(new NIOBlocked(sc2)); 
exec. shutdown() ; 
TimeUnit . SECONDS. sleep(1); 
// Produce an interrupt via cancel: 
f.cancel (true) ; 
TimeUnit . SECONDS. sleep(1); 

. // Release the block by closing the channel: 
sc2.close(); 





} 
} /* Output: (Sample) 
Waiting for read() in NIOBlocked@7a84e4 
Waiting for read() in NIOBlocked@15c785@ 
ClosedByInterruptException 
Exiting NIOBlocked.run() NIOBlocked@15c7850 
AsynchronousCloseException 
Exiting NIOBlocked.run() NIOBlocked@7a84e4 
“i~ 
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如 你 所 见 ， 你 还 可 以 关闭 底层 资源 以 释放 锁 ， 尽 管 这 种 做 法 一 般 不 是 必需 的 。 注 意 ， 使 用 
execute0 来 启动 两 个 任务 ， 并 调用 eshutdownNow0 将 可 以 很 容易 地 终止 所 有 事物 ， 而 对 于 捕获 上 
面 示例 中 的 Future， 只 有 在 将 中 断 发 送 给 一 个 线程 ， 同 时 不 发 送 给 另 一 个 线程 时 才 是 必需 的 9 。 

练习 18: (2) 创建 一 个 非 任务 的 类 ， 它 有 一 个 用 较 长 的 时 间 间 隔 调 用 sleep0 的 方法 。 创 建 一 
个 任务 ， 它 将 调用 这 个 非 任务 类 上 的 那个 方法 。 在 main0 中 ， 启 动 该 任务 ， 然 后 调用 interruptO 
来 终止 它 。 请 确保 这 个 任务 被 安全 地 关闭 。 

练习 19: (4) 修改 OrnamentalGarden.java， 使 其 使 用 interrupt0。 

练习 20: (1) 修改 CachedThreadPooljava， 使 所 有 任务 在 结束 前 都 将 收 到 一 个 interrupt0。 

被 互 斥 所 阻塞 

就 像 在 Interrupting.java 中 看 到 的 ， 如 果 你 尝试 着 在 一 个 对 象 上 调用 其 synchronized 方 法 ， 
而 这 个 对 象 的 锁 已 经 被 其 他 任务 获得 ， 那 么 调用 任务 将 被 挂 起 (阻塞 ) ， 直 至 这 个 锁 可 获得 。 下 
面 的 示例 说 明了 同一 个 互 斥 可 以 如 何 能 被 同一 个 任务 多 次 获得 : 


//: concurrency/MultiLock. java 
// One thread can reacquire the same lock. 
import static net.mindview.util.Print.*; 


public class MultiLock { 
public synchronized void fl(int count) { 
if(count-- > 0) { a 
print("f1() calling 2() with count ”+ count); 
#2(count); 
} 
} 
public synchronized void f2(int count) { 
if(count-- > 6) { 
print("f2() calling f1() with count * + count); 
fi(count): 
} 


public static void main(String(] args) throws Exception { 
final Multilock multiLock = new MultiLock(); 
new Thread() { 
public void run() { 
multiLock. f1(10) ; 


} 
}.start(); 


} 

} /* Output: 

f1() calling f2() with count 9 
#20 calling 1() with count 8 
f10 calling f2() with count 7 
f2() calling f1() with count 6 
f1() calling f2() with count 5 
120 calling f1() with count 4 
f1() calling f2() with count 3 
f2() calling f1() with count 2 
#10 calling f2() with count 1 
2() calling f1() with count 9 
Wm 


在 main0 中 创建 了 一 个 调用 入 0 的 Thread， 然 后 f10 和 f20 互 相 调 用 直至 count 变 为 0。 由 于 这 
个 任务 已 经 在 第 一 个 对 人 0 的 调用 中 获得 了 multiLock 对 象 锁 ， 因 此 同一 个 任务 将 在 对 f20 的 调用 
中 再 次 获取 这 个 锁 ， 依 此 类 推 。 这 么 做 是 有 意义 的 ， 因 为 一 个 任务 应 该 能 够 调用 在 同一 个 对 象 
中 的 其 他 的 synchronized 方 法 ， 而 这 个 任务 已 经 持 有 锁 了 。 


© Ervin Varga 协 助 我 研究 了 本 节 。 
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就 像 前 面 在 不 可 中 断 的 MO 中 所 观察 到 的 那样 ， 无 论 在 任何 时 刻 ， 只 要 任务 以 不 可 中 断 的 方 


式 被 阻塞 ， 那 么 都 有 潜在 的 会 锁 住 程序 的 可 能 。Java SE5 并 发 类 库 中 添加 了 一 个 特性 ， 即 在 
ReentrantLock 上 阻塞 的 任务 具备 可 以 被 中 断 的 能 力 ， 这 与 在 synchronized 方 法 或 临界 区 上 阻塞 
的 任务 完全 不 同 : 


//: concurrency/Interrupting2.java 

// Interrupting a task blocked with a ReentrantLock. 
import java.util.concurrent.*: 

import java.util.concurrent.locks.*: 

import static net.mindview.util.Print.*: 


class BlockedMutex { 
private Lock lock = new ReentrantLock(); 
Public BlockedMutex() { 
// Acquire it right away, to demonstrate interruption 
// of a task blocked on a ReentrantLock: 
lock. lock () ; 
} 
public void fO { 
try { 
// This will never be available to a second task 
lock. lockInterruptibly(); // Special call 
print("lock acquired in f()"); 
} catch(InterruptedException e) { 
print("Interrupted from lock acquisition in f()"); 
} 
2 
} 


class Blocked2 implements Runnable { 
BlockedMutex blocked = new BlockedMutex(); 
public void run() { 
print("Waiting for f() in BlockedMutex"); 
blocked. f () ; 
print("Broken out of blocked call"); 
} 
3 


public class Interrupting? { 
public static void main(String(] args) throws Exception { 
Thread t = new Thread(new Blocked2()) ; 
t.start(); 
TimeUnit SECONDS. sleep(1) ; 
System.out.printin("Issuing t.interrupt()"); 
t.interrupt(); 


} 
} /* Output: 
Waiting for f() in BlockedMutex 
Issuing t.interrupt() 
Interrupted from lock acquisition in f() 
Broken out of blocked call 
Whim 


BlockedMutex 类 有 一 个 构造 器 ， 它 要 获取 所 创建 对 象 上 自身 的 Lock， 并 且 从 不 释放 这 个 锁 。 


出 于 这 个 原因 ， 如 果 你 试图 从 第 二 个 任务 中 调用 f0 (不 同 于 创建 这 个 BlockedMutex 的 任务 )， 那 
么 将 会 总 是 因 Mutex 不 可 获得 而 被 阻塞 。 在 Blocked2 中 ，run0 方 法 总 是 在 调用 blocked.f0 的 地 方 
停止 。 当 运行 这 个 程序 时 ， 你 将 会 看 到 ， 与 1O 调 用 不 同 ，interrupt0 可 以 打 断 被 互 斥 所 阻塞 的 
调用 ®。 


O 注意 ， 尽 管 不 太 可 能 ， 但 是 对 tinterrupt0 的 调用 确实 可 以 发 生 在 对 blocked.f0 的 调用 之 前 。 
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21.4.4 检查 中 断 

注意 , 当 你 在 线程 上 调用 interruptO 时 ,中 断 发 生 的 唯一 时 刻 是 在 任务 要 进入 到 阻塞 操作 中 ， 
或 者 已 经 在 阻塞 操作 内 部 时 《如 你 所 见 ， 除 了 不 可 中 断 的 1O 或 被 阻塞 的 synchronized 方 法 之 外 ， 
在 其 余 的 例外 情况 下 ， 你 无 可 事 事 )。 但 是 如 果 根 据 程序 运行 的 环境 ， 你 已 经 编写 了 可 能 会 产生 
这 种 阻塞 调用 的 代码 ， 那 又 该 怎么 办 呢 ? 如 果 你 只 能 通过 在 阻塞 调用 上 抛 出 异常 来 退出 ， 那 么 
你 就 无 法 总 是 可 以 离开 run0 循 环 。 因此， 如果 你 调用 interrupt0 以 停止 某 个 任务 ， 那 么 在 runO 
循环 碰巧 没有 产生 任何 阻塞 调用 的 情况 下 ， 你 的 任务 将 需要 第 二 种 方式 来 退出 。 

这 种 机 会 是 由 中 断 状 态 来 表示 的 ， 其 状态 可 以 通过 调用 interrupt0 来 设置 。 你 可 以 通过 调用 
interrupted0 来 检查 中 断 状态 ， 这 不 仅 可 以 告诉 你 interrupt0 是 否 被 调用 过 ， 而 且 还 可 以 清除 中 
断 状态 。 清 除 中 断 状态 可 以 确保 并 发 结构 不 会 就 某 个 任务 被 中 断 这 个 问题 通知 你 两 次 ， 你 可 以 
经 由 单一 的 InterruptedException 或 单一 的 成 功 的 Thread.interrupted0 测 试 来 得 到 这 种 通知 。 如 
果 想 要 再 次 检查 以 了 解 是 否 被 中 断 ， 则 可 以 在 调用 Thread.interrupted0 时 将 结果 存储 起 来 。 

下 面 的 示例 展示 了 典型 的 惯用 法 ， 你 应 该 在 run0 方 法 中 使 用 它 来 处 理 在 中 断 状态 被 设置 时 ， 
被 阻塞 和 不 被 阻塞 的 各 种 可 能 : 


//: concurrency/Interrupt ingIdiom. java 
7/ General idiom for interrupting a task. 
// {Args: 1168) 

import java.util.concurrent.*; 

import static net.mindview.util.Print.*; 


class NeedsCleanup { 
private final int id; 
public NeedsCleanup(int ident) { 
id = ident; 
print("NeedsCleanup ”+ id); 


} 
public void cleanup() { 
print("Cleaning up " + id); 
} 
} 


class Blocked3 implements Runnable { 
private volatile double d = 0.0; 
public void run() { 
try { 
while(! Thread. interrupted()) { 
17 pointi 
NeedsCleanup n1 = new NeedsCleanup(1) : 
// Start try-finally immediately after definition 
7/ of mi, to guarantee proper cleanup of n1: 
try { 
print("Sleeping"); 
TimeUnit. SECONDS. sleep(1)+ 
11 point2 
NeedsCleanup n2 = new NeedsCleanup(2) ; 
// Guarantee proper cleanup of n2: 
try { 
print ("Calculating"): 
71 A time-consuming, non-blocking operation: 
for(int i = 1; i < 2500000; i++) 
d= + (Math.PI + Math.E) / d: 
print("Finished time-consuming operation"); 
finally { 
n2.cleanup(): 


finally { 
n.cleanup() ; 


Ld 
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brint CExtting via while() test"); 
} catch(InterruptedException e) { 
print ("Exiting via InterruptedException"); 
} 
} 
} 


public class InterruptingIdiom { 
public static void main(String[] args) throws Exception { 
if(args.length != 1) { 
print("usage: java InterruptingIdiom delay-in-mS"); 
System.exit(1): 


2 

Thread t = new Thread(new Blocked3()); 

t.start(); 

TimeUnit MILLISECONDS. sleep(new Integer (args(0])): 
t.interrupt(); 


} he Output: (Sample) 

NeedsCleanup 1 

Sleeping 

NeedsCleanup 2 

Calculating 

Finished time-consuming operation 

Cleaning up 2 

Cleaning up 1 

NeedsCleanup 1 

Sleeping 

Cleaning up 1 

Exiting via InterruptedException 

AAA ~ 

NeedsCleanup 类 强调 在 你 经 由 异常 离开 循环 时 ， 正 确 清理 资源 的 必要 性 。 注 意 ， 所 有 在 
Blocked3.run0 中 创建 的 NeedsCleanup 资 源 都 必须 在 其 后 面 紧 跟 try-finally 子 句 ， 以 确保 cleanup0O 
方法 总 是 会 被 调用 。 

你 必须 给 程序 提供 一 个 命令 行 参数 ， 来 表示 在 它 调用 interrupt0 之 前 以 毫秒 为 单位 的 延迟 时 
间 。 通 过 使 用 不 同 的 延迟 ， 你 可 以 在 不 同 地 点 退出 Blocked3.run0: 在 阻塞 的 sleepO 调 用 中 ， 或 
者 在 非 阻塞 的 数学 计算 中 。 你 将 看 到 ， 如 果 interrupt0 在 注释 point2 之 后 ( 即 在 非 阻塞 的 操作 过 
程 中 ) 被 调用 ， 那 么 首先 循环 将 结束 ， 然 后 所 有 的 本 地 对 象 将 被 销毁 ， 最 后 循环 会 经 由 while 语 
名 的 顶部 退出 。 但 是 ， 如 果 interruptO 在 pointl 和 point2 之 间 (在 while 语 句 之 后 ， 但 是 在 阻塞 操 
作 sleep0 之 前 或 其 过 程 中 ) 被 调用 ， 那 么 这 个 任务 就 会 在 第 一 次 试图 调用 阻塞 操作 之 前 ， 经 由 
InterruptedException 退 出 。 在 这 种 情况 下 ， 在 异常 被 抛 出 之 时 唯一 被 创建 出 来 的 NeedsCleanup 
对 象 将 被 清除 ， 而 你 也 就 有 了 在 catch 子 句 中 执行 其 他 任何 清除 工作 的 机 会 。 

被 设计 用 来 响应 interrupt0 的 类 必须 建立 一 种 策略 ， 来 确保 它 将 保持 一 致 的 状态 。 这 通常 意 
味 着 所 有 需要 清理 的 对 象 创建 操作 的 后 面 ， 都 必须 紧 跟 try-finally 子 句 ， 从 而 使 得 无 论 run0 循 环 
如 何 退 出 ， 清 理 都 会 发 生 。 像 这 样 的 代码 会 工作 得 很 好 ， 但 是 ， 唉 ， 由 于 在 Java 中 缺乏 自动 的 
析 构 器 调用 ， 因 此 这 将 依赖 于 客户 端 程序 员 去 编写 正确 的 try-finally 子 句 。 


21.5 线程 之 间 的 协作 


正如 你 所 见 到 的 ， 当 你 使 用 线程 来 同时 运行 多 个 任务 时 ， 可 以 通过 使 用 锁 (ER) 来 同步 
两 个 任务 的 行为 ， 从 而 使 得 一 个 任务 不 会 干涉 另 一 个 任务 的 资源 。 也 就 是 说 ， 如 果 两 个 任务 在 
交替 着 步 人 某 项 共享 资源 (通常 是 内 存 ) ， 你 可 以 使 用 互 斥 来 使 得 任何 时 刻 只 有 一 个 任务 可 以 访 
问 这 项 资源 。 





ray) 
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这 个 问题 已 经 解决 了 ， 下 一 步 是 学 习 如 何 使 任务 彼此 之 间 可 以 协作 ， 以 使 得 多 个 任务 可 以 
一 起 工作 去 解决 某 个 问题 。 现 在 的 问题 不 是 彼此 之 间 的 干涉 ， 而 是 彼此 之 间 的 协调 ， 因 为 在 这 
类 问题 中 ， 某 些 部 分 必须 在 其 他 部 分 被 解决 之 前 解决 。 这 非常 像 项 目 规 划 ， 必 须 先 挖 房子 的 地 
基 ， 但 是 接 下 来 可 以 并 行 地 铺设 钢 结构 和 构建 水 泥 部 件 ， 而 这 两 项 任务 必须 在 混凝土 浇注 之 前 
完成 。 管 道 必须 在 水 泥 板 浇注 之 前 到 位 ， 而 水 泥 板 必须 在 开始 构筑 房屋 骨架 之 前 到 位 ， 等 等 。 
在 这 些 任务 中 ， 某 些 可 以 并 行 执行 ， 但 是 某 些 步骤 需要 所 有 的 任务 都 结束 之 后 才能 开动 。. 

当 任务 协作 时 ， 关 键 问 题 是 这 些 任务 之 间 的 担 手 。 为 了 实现 这 种 担 手 ， 我 们 使 用 了 相同 的 
基础 特性 : 互 斥 。 在 这 种 情况 下 ， 互 斥 能 够 确保 只 有 一 个 任务 可 以 响应 某 个 信号 ， 这 样 就 可 以 
根除 任何 可 能 的 竞争 条 件 。 在 互 斥 之 上 ， 我 们 为 任务 添加 了 一 种 途径 ， 可 以 将 其 自身 挂 起 , 直 
至 某 些 外 部 条 件 发 生变 化 (例如 , 管道 现在 已 经 到 位 )， 表示 是 时 候 让 这 个 任务 向 前 开动 了 为 止 。 
在 本 节 ， 我 们 将 浏览 任务 间 的 担 手 问题 ， 这 种 担 手 可 以 通过 Object 的 方法 wait0 和 motify0 来 安全 
地 实现 。Java SE5 的 并 发 类 库 还 提供 了 具有 await0 和 signal0 方 法 的 Condition 对 象 。 我 们 将 看 到 
产生 的 各 类 问题 ， 以 及 相应 的 解决 方案 。 

21.5.1 wait()4jnotifyAll() 

wait0 使 你 可 以 等 待 菜 个 条 件 发 生变 化 ,而 改变 这 个 条 件 超出 了 当前 方法 的 控制 能 力 。 通 常 ， 
这 种 条 件 将 由 另 一 个 任务 来 改变 。 你 肯定 不 想 在 你 的 任务 测试 这 个 条 件 的 同时 ， 不 断 地 进行 空 
循环 ， 这 被 称 为 忙 等 待 ， 通 常 是 一 种 不 良 的 CPU 周期 使 用 方式 。 因 此 wait0 会 在 等 待 外 部 世界 产 
生变 化 的 时 候 将 任务 挂 起 ， 并 且 只 有 在 notify0 或 notifyANO 发 生 时 ， 即 表示 发 生 了 某 些 感 兴趣 的 
事物 ， 这 个 任务 才 会 被 唤醒 并 去 检查 所 产生 的 变化 。 因 此 ，wait0 提 供 了 一 种 在 任务 之 间 对 活动 
同步 的 方式 。 

调用 sleep0 的 时 候 锁 并 没有 被 释放 ， 调 用 yield0 也 属于 这 种 情况 ， 理 解 这 一 点 很 重要 。 另 一 
方面 ， 当 一 个 任务 在 方法 里 遇 到 了 对 wait0 的 调用 的 时 候 ， 线 程 的 执行 被 挂 起 ， 对 象 上 的 锁 被 释 
wait0 将 释放 锁 ， 这 就 意味 着 另 一 个 任务 可 以 获得 这 个 锁 , 因此 在 该 对 象 (现在 是 未 锁 
定 的 ) 中 的 其 他 synchronized 方 法 可 以 在 waitO 期 间 被 调用 。 这 一 点 至 关 重要 ， 这 些 其 他 的 
方法 通常 将 会 产生 改变 ， 而 这 种 改变 正 是 使 被 挂 起 的 任务 重新 唤醒 所 感 兴趣 的 变化 。 因 此 ， 当 
你 调用 wait0 时 ， 就 是 在 声明 :“ 我 已 经 刚刚 做 完 能 做 的 所 有 事情 ， 因 此 我 要 在 这 里 等 待 ， 但 是 
我 希望 其 他 的 synchronized 操 作 在 条 件 适 合 的 情况 下 能 够 执行 。 

有 两 种 形式 的 wait0。 第 一 种 版 本 接受 毫秒 数 作为 参数 ， 含 义 与 sleep0 方 法 里 参数 的 意思 相 
同 ， 都 是 指 “ 在 此 期 间 暂 停 "。 但 是 与 sleep0 不 同 的 是 ， 对 于 wait0 而 言 : 

1) 在 wait0 期 间 对 象 锁 是 释放 的 。 X 

2) 可 以 通过 notify0、notifyAll0， 或 者 令 时 间 到 期 ， 从 wait0 中 恢复 执行 。 

第 二 种 ， 也 是 更 常用 形式 的 wait0 不 接受 任何 参数 。 这 种 wait0 将 无 限 等 待 下 去 ， 直 到 线程 
接收 到 notify0 或 者 notifyAHO 消 息 。 

wait0、notify0 以 及 notifyAlIO 有 一 个 比较 特殊 的 方面 ， 那 就 是 这 些 方法 是 基 类 Object 的 一 
部 分 ， 而 不 是 属于 Thread 的 一 部 分 。 尽 管 开始 看 起 来 有 点 奇怪 一 一 仅仅 针对 线程 的 功能 却 作为 
通用 基 类 的 一 部 分 而 实现 ， 不 过 这 是 有 道理 的 ， 因 为 这 些 方法 操作 的 锁 也 是 所 有 对 象 的 一 部 分 。 
所 以 ， 你 可 以 把 wait0 放 进 任何 同步 控制 方法 里 ， 而 不 用 考虑 这 个 类 是 继承 自 Thread 还 是 实现 了 
Runnable 接 口 。 实 际 上 ， 只 能 在 同步 控制 方法 或 同步 控制 块 里 调用 wait0、notifyO 和 notifyAli0 
(因为 不 用 操作 锁 ， 所 以 sleep0 可 以 在 非 同 步 控制 方法 里 调用 )。 如 果 在 非 同步 控制 方法 里 调用 
这 些 方法 ， 程 序 能 通过 编译 ， 但 运行 的 时 候 ， 将 得 到 TllegalMonitorStateException 异 常 ， 并 伴随 
着 一 些 含糊 的 消息 ， 比 如 “当前 线程 不 是 拥有 者 "。 消 息 的 意思 是 ， 调 用 wait0、notify0 和 











1197} 
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notifyAI0 的 任务 在 调用 这 些 方法 前 必须 “拥有 ”( 获 取 ) 对 象 的 锁 。 
可 以 让 另 一 个 对 象 执行 某 种 操作 以 维护 其 自己 的 锁 。 要 这 么 做 的 话 ， 必 须 首先 得 到 对 象 的 
锁 。 比 如 ， 如 果 要 向 对 象 x 发 送 notifyAlI0， 那 么 就 必须 在 能 够 取得 x 的 锁 的 同步 控制 块 中 这 么 做 ; 


synchronized(x) { 
x.notifyALl() ; 
) 


让 我 们 看 一 个 简单 的 示例 ，WaxOMaticjava 有 两 个 过 程 : 一 个 是 将 蜡 涂 到 Car 上 ， 一 个 是 
抛光 它 。 抛 光 任务 在 涂 蜡 任务 完成 之 前 ， 是 不 能 执行 其 工作 的 ， 而 涂 蜡 任务 在 涂 另 一 层 蜡 之 前 ， 
必须 等 待 抛光 任务 完成 。WaxOn 和 WaxOff 都 使 用 了 Car 对 象 ， 该 对 象 在 这 些 任务 等 待 条 件 变 化 
的 时 候 ， 使 用 wait0 和 notifyAlI0 来 挂 起 和 重新 启动 这 些 任务 ; 


11: concurrency/waxomat ic/WaxOMat ic. java 
// Basic task cooperation. 
package concurrency. waxomatic: 
import jaVa.util.concurrent.*; 
import static net.mindview.util.Print.*; 
class Car { 
private boolean waxOn = false: 
public synchronized void waxed() { 
waxOn = true; // Ready to buff 
notifyAll (): 


7 

public synchronized void buffed() { 
waxOn = false; // Ready for another coat of wax 
notifyAll (); 


public synchronized void waitForWaxing() 
throws InterruptedException { 
while(waxOn == false) 
wait(); 
>? 
public synchronized void waitForBuffing() 
throws InterruptedException { 
while(waxOn == true) 
wait(); 
} 


} 


class WaxOn implements Runnable { 

private Car car; 

public WaxOn(Car c) { car = c; } 

public void run() { 

try { 
while(!Thread.interrupted()) { 

printnb("Wax On! “): 
TimeUnit MILLISECONDS. sleep(2@8) ; 
car.waxed() ; 
car.waitForBuffing(); 


} 
} catch(InterruptedException e) { 
print("Exiting via interrupt"); 


} 
print("Ending Wax On task"); 


} 


+ class WaxOff implements Runnable { 
private Car car; 
public WaxOff(Car c) { car = c; } 
public void run() { 
try { 
~ while(!Thread. interrupted()) { 
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car .waitForWaxing(); 
printnb("Wax Off! "); 

TimeUnit MILLISECONDS. sleep(200) ; 
car.buffed(); 


} catch(InterruptedException e) { 
print("Exiting via interrupt”): 


} 
print("Ending Wax Off task"); 
} 
} 


public class WaxOMatic { y 
public static void main(String[] args) throws Exception { 
Car car = new Car(); 
ExecutorService exec = Executors.newCachedThreadPool () ; 
exec.execute(new WaxOff (car)): 
exec.execute(new WaxOn(car)); 
TimeUnit.SECONDS.sleep(5); // Run for a while... 
exec.shutdownNow(); // Interrupt all tasks 


} 
} /* Output: (95% match) 
Wax On! Wax Off! Wax On! Wax Off! Wax On! Wax Off! Wax On! 
Wax Off! Wax On! Wax Off! Wax On! Wax Off! Wax On! Wax Off! 
Wax On! Wax Off! Wax On! Wax Off! Wax On! Wax Off! Wax On! 
Wax Off! Wax On! Wax Off! Wax On! Exiting via interrupt 
Ending Wax On task 
Exiting via interrupt 
Ending Wax Off task 
Wh 


这 里 ，Car 有 一 个 单一 的 布尔 属性 waxOn， 表 示 涂 蜡 - 抛 光 处 理 的 状态 。 

在 waitForWaxing0 中 将 检查 waxOn 标 志 ， 如 果 它 为 false， 那 么 这 个 调用 任务 将 通过 调用 
waitO 而 被 挂 起 。 这 个 行为 发 生 在 synchronized 方 法 中 这 一 点 很 重要 ， 因 为 在 这 样 的 方法 中 ， 任 
务 已 经 获得 了 锁 。 当 你 调用 waitO 时 ， 线 程 被 挂 起 ， 而 锁 被 释放 。 锁 被 释放 这 一 点 是 本 质 所 在 ， 
因为 为 了 安全 地 改变 对 象 的 状态 〈 例 如 ， 将 waxOn 改 变 为 true， 如 果 被 挂 起 的 任务 要 继续 执行 ， 
就 必须 执行 该 动作 ) ， 其 他 某 个 任务 就 必须 能 够 获得 这 个 锁 。 在 本 例 中 ， 如 果 另 一 个 任务 调用 
waxed0 来 表示 “是 时 候 该 二 点 什么 了 "， 那 么 就 必须 获得 这 个 锁 ， 从 而 将 waxOn 改 变 为 true。 之 
后 ，waxed(O) 调 用 notifyAUO， 这 将 唤醒 在 对 wait(O) 的 调用 中 被 挂 起 的 任务 。 为 了 使 该 任务 从 
wait0 中 唤醒 ， 它 必须 首先 重新 获得 当 它 进入 wait0 时 释放 的 锁 。 在 这 个 锁 变 得 可 用 之 前 ， 这 个 
任务 是 不 会 被 唤醒 的 。。 

WaxOn.run0 表 示 给 汽车 打 蜡 过 程 的 第 一 个 步骤 ， 因 此 它 将 执行 它 的 操作 : 调用 sleepO 以 模 
拟 需要 涂 蜡 的 时 间 ， 然 后 告知 汽车 涂 蜡 结 束 ， 并 调用 waitForBuffing0， 这 个 方法 会 用 一 个 wait0 
调用 来 挂 起 这 个 任务 ， 直 至 WaxOff 任 务 调用 这 辆 车 的 buffed0， -从 而 改变 状态 并 调用 niotifyAlO0 
为 止 。 另 一 方面 ，WaxO 人 frun0 立 即 进入 waitForWaxing0， 并 因此 而 被 挂 起 ， 直 至 WaxOn 涂 完 
蜡 并 且 waxed0 被 调用 。 在 运行 这 个 程序 时 ， 你 可 以 看 到 当 控制 权 在 两 个 任务 之 间 来 回 互相 传递 
时 ， 这 个 两 步骤 过 程 在 不 断 地 重复 。 在 5 秒 钟 之 后 ，interruiptO 会 中 止 这 两 个 线程 ， 当 你 调用 某 
个 ExecutorService 的 shutdownNow0 时 ， 它 会 调用 所 有 由 它 控制 的 线程 的 interruptO 。 


O 在 某 些 平台 上 还 有 第 三 种 从 wait0 中 抽身 而 出 的 方式 ， 即 所 谓 的 伪 吹 醒 。 伪 唤醒 实质 上 意味 着 一 个 线程 (在 等 
待 某 个 条 件 变量 或 信号 量 时 ) 可 以 过 早 地 停止 阻塞 ， 而 不 需要 由 notify0 或 aotifyANO (或 者 与 它们 等 价 的 新 的 
Condition 对 象 ) 来 提示 。 这 个 线程 表面 上 看 起 来 是 由 其 自身 允 醒 的 。 伪 唤醒 之 所 以 存在 ， 是 因为 实现 POSIX 
线程 ， 或 者 其 等 价 物 ， 在 某 些 平台 上 ， 并 非 4 总 是 如 它们 应 该 表现 出 的 那样 简单 直观 。 伪 唤 醒 机 制 使 得 在 这 些 平 
台 .上 执行 诸如 构建 像 pthreads 这 样 的 类 库 的 工作 会 容易 一 些 。 
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前 面 的 示例 强调 你 必须 用 一 个 检查 感 兴趣 的 条 件 的 while 循 环 包围 wait0。 这 很 重要 ， 因 为 : 
“你 可 能 有 多 个 任务 出 于 相同 的 原因 在 等 待 同 一 个 锁 ， 而 第 一 个 唤醒 任务 可 能 会 改变 这 种 状 

1202] 况 ( 即 使 你 没有 这 么 做 ， 有 人 也 会 通过 继承 你 的 类 去 这 么 做 ) 。 如 果 属 于 这 种 情况 ， 那 么 
这 个 任务 应 该 被 再 次 挂 起 ， 直 至 其 感 兴趣 的 条 件 发 生变 化 。 

“在 这 个 任务 从 其 wait0 中 被 唤醒 的 时 刻 ， 有 可 能 会 有 某 个 其 他 的 任务 已 经 做 出 了 改变 ， 从 
而 使 得 这 个 任务 在 此 时 不 能 执行 ， 或 者 执行 其 操作 已 显得 无 关 紧 要 。 此 时 ， 应 该 通过 再 次 
调用 wait0 来 将 其 重新 挂 起 。 

* 也 有 可 能 某 些 任务 出 于 不 同 的 原因 在 等 待 你 的 对 象 上 的 锁 (在 这 种 情况 下 必须 使 用 
notifyAll0)。 在 这 种 情况 下 ， 你 需要 检查 是 否 已 经 由 正确 的 原因 唤醒 ， 如 果 不 是 ， 就 再 次 
调用 wait0。 s 
因此 ， 其 本 质 就 是 要 检查 所 感 兴趣 的 特定 条 件 ， 并 在 条 件 不 满足 的 情况 下 返回 到 wait0 中 。 

惯用 的 方法 就 是 使 用 while 来 编写 这 种 代码 。 

练习 21: (2) 创建 两 个 Runnable， 其 中 一 个 的 run0 方 法 启动 并 调用 wait0， 而 第 二 个 类 应 该 

捕获 第 一 个 Runnable 对 象 的 引用 ， 其 ran() 方 法 应 该 在 一 定 的 秒 数 之 后 ， 为 第 一 个 任务 调用 
motifyAIO， 从 而 使 得 第 一 个 任务 可 以 显示 一 条 信息 。 使 用 Executor 来 测试 你 的 类 。 

练习 22: (4) 创建 一 个 忙 等 待 的 示例 。 第 一 个 任务 休眠 一 段 时 间 然 后 将 一 个 标志 设置 为 true， 

而 第 二 个 任务 在 一 个 while 循 环 中 观察 这 个 标志 (这 就 是 忙 等 待 )， 并 且 当 该 标志 变 为 true 时 ， 将 
其 设置 回 false， 然 后 向 控制 台 报告 这 个 变化 。 请 注意 程序 在 忙 等 待 中 浪费 了 多 少时 间 ， 然 后 创 
建 该 程序 的 第 二 个 版 本 ， 其 中 将 使 用 wait0 而 不 是 忙 等 待 。 

错失 的 信号 

当 两 个 线程 使 用 notify0/wait0 或 notifyAlI0/wait0 进 行 协作 时 ， 有 可 能 会 错过 某 个 信号 。 假 
设 T1 是 通知 T2 的 线程 ， 而 这 两 个 线程 都 是 使 用 下 面 有 缺陷 的 ) 方式 实现 的 : 


m 
synchronized(sharedMonitor) ¢ 
<setup condition for T2> 
sharedMonitor.notify(); 














k 


white (soneCondst ton) { 
// Point 1 
1203 synchrontzed(sharedMonitor) { 
sharedMonitor.wait(); 

} 

<Setup condition for T2> 是 防止 T2 调 用 wait0 的 一 个 动作 ， 当 然 前 提 是 T2 还 没有 调用 wait0 。 

假设 T2 对 someCondition 求 值 并 发 现 其 为 true。 在 Pointl， 线 程 调度 器 可 能 切换 到 了 T1。 而 
T1 将 执行 其 设置 ， 然 后 调用 notify0。 当 T2 得 以 继续 执行 时 ,此 时 对 于 T2 来 说 , 时 机 已 经 太 晚 了 ， 
以 至 于 不 能 意识 到 这 个 条 件 已 经 发 生 了 变化 ， 因 此 会 定 目 进入 wait0。 此 时 notify0 将 错失 ， 而 
T2 也 将 无 限 地 等 待 这 个 已 经 发 送 过 的 信号 ， 从 而 产生 死 锁 。 

该 问题 的 解决 方案 是 防止 在 someConditon 变 量 上 产生 竞争 条 件 。 下 面 是 T2 正 确 的 执行 方式 : 


synchronized(sharedMonitor) { 
1 while(someCondition) 
| sharedMonitor.wait(); 

} 


现在 ， 如 果 T1 首 先 执行 ， 当 控制 返回 T2 时 ， 它 将 发 现 条 件 发 生 了 变化 ， 从 而 不 会 进入 wait0。 
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反 过 来 ， 如 果 T2 首 先 执行 ， 那 它 将 进入 waitO0， 并 且 稍 后 会 由 TI 唤醒 。 因 此 ， 信 号 不 会 错失 。 
21.5.2 notify() 与 notifyAll() 

因为 在 技术 上 ， 可 能 会 有 多 个 任务 在 单个 Car 对 象 上 处 于 wait0 状 态 ， 因 此 调用 notifyAHO 比 
只 调用 notify0 要 更 安全 。 但 是 ， 上 面 程序 的 结构 只 会 有 一 个 任务 实际 处 于 wait0 状 态 ， 因 此 你 可 
以 使 用 notify0 来 代替 notifyAlIO。 

使 用 notify0O 而 不 是 notifyANO 是 一 种 优化 。 使 用 notify0 时 ， 在 众多 等 待 同一 个 锁 的 任务 中 
只 有 一 个 会 被 唤醒 ， 因 此 如 果 你 希望 使 用 notifyg0， 就 必须 保证 被 唤醒 的 是 恰当 的 任务 。 另 外 ， 
为 了 使 用 notifty0， 所 有 任务 必须 等 待 相同 的 条 件 ， 因 为 如 果 你 有 多 个 任务 在 等 待 不 同 的 条 件 ， 
那么 你 就 不 会 知道 是 否 唤醒 了 恰当 的 任务 。 如 果 使 用 notify0， 当 条 件 发 生变 化 时 ， 必 须 只 有 一 
个 任务 能 够 从 中 受益 。 最 后 ， 这 些 限制 对 所 有 可 能 存在 的 子 类 都 必须 总 是 起 作用 的 。 如 果 这 些 
规则 中 有 任何 一 条 不 满足 ， 那 么 你 就 必须 使 用 notifyAlI0 而 不 是 notify0。 1204 

在 有 关 Java 的 线程 机 制 的 讨论 中 ， 有 一 个 令 人 困惑 的 描述 : notifyAl10 将 唤醒 “所 有 正在 等 
待 的 任务 "。 这 是 否 意味 着 在 程序 中 任何 地 方 ， 任 何 处 于 wait() 状 态 中 的 任务 都 将 被 任何 对 
notifyAll0 的 调用 唤醒 呢 ? 在 下 面 的 示例 中 ， 与 Task2 相 关 的 代码 说 明了 情况 并 非 如 此 一 一 事实 
上 ， 当 notifyAlIO 因 某 个 特定 锁 而 被 调用 时 ， 只 有 等 待 这 个 锁 的 任务 才 会 被 唤醒 ; 


//: concurrency/NotifyVsNotifyALl. java 
import java.util.concurrent.*; 
import java.util.*; 














class Blocker { 
synchronized void waitingCatl() { 
try { 
while(! Thread. interrupted()) { 
wait(); 
System.out.print(Thread.currentThread() + " "); 


} catch(InterruptedException e) { 
/1 OK to exit this way 
} 
} 
synchronized void prod() { notify(); } 
synchronized void prodAll() { notifyAll(); } 
} 


class Task implements Runnable { 
static Blocker blocker = new Blocker(); 
public void run() { blocker.waitingCall(); } 
} 


class Task2 implements Runnable { 
// A separate Blocker object: 
static Blocker blocker = new Blocker(); 
public void run() { blocker.waitingCall(): } 
} 


public class NotifyVsNotifyAll { 
public static void main(String[] args) throws Exception { 
ExecutorService exec = Executors.newCachedThreadPool () ; 
for(int i = 0; i < 5; i++) 
exec.execute(new Task()): 
exec.execute(new Task2()); 
Timer timer = new Timer(); 1205] 
timer .scheduleatFixedRate(new TimerTask() { 
boolean prod = true; 
public void run() { 
if(prod) { 
System. out.print("\nnotify() "); 
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Task. blocker .prod(); 
prod = false; 
} else { 
System.out.print("\nnotifyAll() "); 
Task. blocker .prodAl1(); 
prod = true; 
} 
} 
}, 480, 400); // Run every .4 second 
TimeUnit. SECONDS. steep(5); // Run for a while... 
timer.cancel(); 
System.out.printin("\nTimer canceled"); 
TimeUnit MILLISECONDS . sleep (590) ; 
System.out.print("Task2.blocker.prodAll() "); 
Task2. blocker .prodAll(): 
TimeUnit . MILLISECONDS. sleep(500) ; 
System. out.printin("\nShutting down"); 
exec. shutdownNow(); // Interrupt all tasks 
} 
} /* Output: (Sample) 
notify() Thread{pool-1-thread-1,5,main] 
notifyAll() Thread[pool-1-thread-1,5,main] Thread[pool-1- 
thread-5,5,main] Thread{pool-1-thread-4,5,main] 
Thread{pool-1-thread-3,5,main) Thread{pool-1-thread- 
2,5.main} 
notify() Thread(pool-1-thread-1,5,main} 
notifyAll() Thread(pool-1-thread-1,5,main] Thread{pool-1- 
thread-2,5,main] Thread(pool-1-thread-3,5,main} 
Thread(pool-1-thread-4,5,main} Thread{pool-i-thread- 
5,5,main] 
notify() Thread(pool-1-thread-1,5,main] 
notifyAll() Thread{pool-1-thread-1,5,main] Thread(pool-1- 
thread-5,5,main] Thread[{pool-1-thread-4,5,main} 
Thread{pool-1-thread-3,5,main] Thread{pool-1-thread- 
2,5,main] 
notify() Thread{pool-1-thread-1,5,main] 
notifyAll() Thread[(pool-1-thread-1,5,main] Thread[poot-1- 
thread-2,5,main] Thread{pool-1-thread-3,5,main] 
Thread{pool-1-thread-4,5,main] Thread{pool-1-thread- 
5,5,main) 
notify() Thread[pool-1-thread-1,5,main] 
notifyAll() Thread[{pool-1-thread-1,5,main] Thread[pool-1- 
thread-5,5,main] Thread{pool-1-thread-4,5,main} 
Thread{pool-1-thread-3,5,main] Thread[pool-1-thread- 
2,5,main] 
notify() Thread[poot-1-thread-1,5,main] 
notifyAll() Thread[pool-1-thread-1,5,main] Thread[pool-1- 
thread-2,5,main] Thread{pool-1-thread-3,5,main] 
Threadlpool-1-thread-4,5.main] Thread[pool-1-thread- 
5,5,main] 
Timer canceled 
Task2.blocker.prodAll() Thread{pool-1-thread-6,5,mainj 
Shutting down 
2 


Task 和 Task2 每 个 都 有 其 自己 的 Blocker 对 象 ， 因 此 每 个 Task 对 象 都 会 在 Task.blocker 上 阻塞 ， 
而 每 个 Task2 都 会 在 Task2.blocker 上 阳 塞 。 在 main0 中 ，java.util.Timer 对 象 被 设置 为 每 4/10 秒 执 
行 一 次 run0 方 法 ， 而 这 个 ran0 方 法 将 经 由 “激励 ”方法 交替 地 在 Task.blocker 上 调用 notify0 和 
notifyAll, 

从 输出 中 你 可 以 看 到 ， 即 使 存在 Task2.blocker 上 阻塞 的 Task2 对 象 ， 也 没有 任何 在 
Task-blocker 上 的 notify0 或 notifyAIIO 调 用 会 导致 ask2 对 象 被 唤醒 。 与 此 类 似 ， 在 main0 的 结尾 ， 
调用 了 timer 的 cancel0， 即 使 计时 器 被 撤销 了 ， 前 5 个 任务 也 依然 在 运行 ， 并 仍旧 在 它们 对 
Task.blockerwaitingCall0 的 调用 中 被 阻塞 。 对 Task2.blockerprodAll0 的 调用 所 产生 的 输出 不 包括 
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任何 在 Task.blocker 中 的 锁 上 等 待 的 任务 。 

如 果 你 浏览 Blocker 中 的 prod() 和 prodAll()， 就 会 发 现 这 是 有 意义 的 。 这 些 方 法 是 
synchronized 的 ， 这 意味 着 它们 将 获取 自身 的 锁 ， 因 此 当 它 们 调用 notify0 或 notifyAlI0 时 ， 只 在 
这 个 锁 上 调用 是 符合 逻辑 的 一 因此 ， 将 只 唤醒 在 等 待 这 个 特定 锁 的 任务 。 

Blocker.waitingCall0 非 常 简单 ， 以 至 于 在 本 例 中 ， 你 只 需 声 明 for(;;) 而 不 是 while(!Thread. 
interrupted()) 就 可 以 达到 相同 的 效果 ， 因 为 在 本 例 中 ， 由 于 异常 而 离开 循环 和 通过 检查 
interruptedO 标 志 离 开 循环 是 没有 任何 区 别 的 一 在 两 种 情况 下 都 要 执行 相同 的 代码 。 但 是 , 事 
实 上 ， 这 个 示例 选择 了 检查 interrupted0， 因 为 存在 着 两 种 离开 循环 的 方式 。 如 果 在 以 后 的 某 个 
时 刻 ， 你 决定 要 在 循环 中 添加 更 多 的 代码 ， 那 么 如 果 没 有 覆盖 从 这 个 循环 中 退出 的 这 两 条 路 径 ， 
就 会 产生 引入 错误 的 风险 。 

练习 23，(7) 演示 当 你 使 用 notify0 来 代替 notifyAlI0 时 ，WaxOMatic.java 可 以 成 功 地 工作 。 
21.5.3 生产 者 与 消费 者 

请 考虑 这 样 一 个 饭店 ， 它 有 一 个 厨师 和 一 个 服务 员 。 这 个 服务 员 必 须 等 待 厨师 准备 好 腾 食 。 
当 厨 师 准备 好 时 ， 他 会 通知 服务 员 ， 之 后 服务 员 上 菜 ， 然 后 返回 继续 等 待 。 这 是 一 个 任务 协作 
的 示例 厨师 代表 生产 者 ， 而 服务 员 代 表 消费 者 。 两 个 任务 必须 在 腾 食 被 生产 和 消费 时 进行 握 
手 ， 而 系统 必须 以 有 序 的 方式 关闭 。 下 面 是 对 这 个 叙述 建 模 的 代码 ， 


//:_concurrency/Restaurant. java 
// The producer-consumer approach to task cooperation. 
import java.util.concurrent.*; 

import static net.mindview.util.Print.*; 


class Meal { 
private final int orderNum; 
public Meal (int orderNum) { this.orderNum = orderNum; } 
public String toString() { return “Meal ”+ orderNum: } 
} 


class WaitPerson implements Runnable { 

private Restaurant restaurant; 

public WaitPerson(Restaurant r) { restaurant = r; } 
public void run() { 

try { 
while(!Thread.interrupted()) { 
synchronized(this) { 
while(restaurant.meal == null) 
wait(); // ... for the chef to produce a meal 


$ 
print("Waitperson got ”+ restaurant .meal); 
synchronized(restaurant.chef) { 

restaurant.meal = null; 

restaurant.chef .notifyAll(); // Ready for another 
4 


} P 
} catch(InterruptedException e) { 
print("WaitPerson interrupted"): 
} 
} 
} 


class Chef implements Runnable { 
private Restaurant restaurant; 
private int count = @; 
public Chef (Restaurant r) { restaurant = r; } 
public void run() { 
try { 
while(!Thread. interrupted()) { 
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synchronized(this) { 
while(restaurant.meal != null) 
wait(); // ... for the meal to be taken 


} 

if(++count == 16) { 
print("Out of food, closing"); 
restaurant exec. shutdownNow() ; 


printnb("Order up! "); ， 
synchronized(restaurant.waitPerson) { 
restaurant.meal = new Meal (count); 
restaurant.waitPerson.notifyAll(); 


} 
TimeUnit MILLISECONDS. sleep(16@) ; 


} 
} catch(InterruptedException e) { 
print("Chef interrupted"); 


} 
} 


public class Restaurant { 
Meal meal; 
ExecutorService exec = Executors.newCachedThreadPool () ; 
WaitPerson waitPerson = new WaitPerson(this); 
Chef chef = new Chef (this); 
public Restaurant() { 
exec. execute (chef); 
exec. execute(waitPerson) ; 


} 
public static void main(String[] args) { 
new Restaurant(); 


} 
} /* Output: 
Order up! Waitperson got Meal 
Order up! Waitperson got Meal 
Order up! Waitperson got Meal 
Order up! Waitperson got Meal 
Order up! Waitperson got Meal 
Order up! Waitperson got Meal 
Order up! Waitperson got Meal 
Order up! Waitperson got Meal 
Order up! Waitperson got Meal 
Out of food, closing 
WaitPerson interrupted 
Order up! Chef interrupted 
Whim 


Restaurant 是 WaitPerson 和 Chef 的 焦点 ， 他 们 都 必须 知道 在 为 哪个 Restaurant 工 作 ， 因 为 他 
们 必须 和 这 家 饭店 的 “ 餐 窗 ”打交道 ， 以 便 放置 或 拿 取 腾 食 restaurant.meal。 在 run() 中 ， 
WaitPerson 进 入 wait0 模 式 ， 停 止 其 任务 ， 直 至 被 Chef 的 notifyANIO 唤 醒 。 由 于 这 是 一 个 非常 简 
单 的 程序 ， 因 此 我 们 知道 只 有 一 个 任务 将 在 WaitPerson 的 锁 上 等 待 : 即 WaitPerson 任 务 自身 。 
出 于 这 个 理论 上 可 以 调用 notify0 而 不 是 notifyAl1I0。 但 是 ， 在 更 复杂 的 情况 下 ， 可 能 会 有 
多 个 任务 在 某 个 特定 对 象 锁 上 等 待 ， 因 此 你 不 知道 哪个 任务 应 该 被 唤醒 。 因 此 ， 调 用 notifyAlIO 
要 更 安全 一 些 ， 这 样 可 以 唤醒 等 待 这 个 锁 的 所 有 任务 ， 而 每 个 任务 都 必须 决定 这 个 通知 是 否 与 
自己 相关 。 

一 旦 Chef 送 上 Meal 并 通知 WaitPerson， 这 个 Chef 就 将 等 待 ， 直 至 WaitPerson 收 集 到 订单 并 
通知 Chef， 之 后 Chef 就 可 以 烧 下 一 份 Meal 了 。 

注意 ，wait0 被 包装 在 一 个 while0 语 句 中 ， 这 个 语句 在 不 断 地 测试 正在 等 待 的 事物 。 告 看 上 
去 这 有 点 怪 一 如 果 在 等 待 一 个 订单 ， 一 旦 你 被 唤醒 ， 这 个 订单 就 必定 是 可 获得 的 ， 对 吗 ? E 
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如 前 面 注意 到 的 ， 问 题 是 在 并 发 应 用 中 ， 某 个 其 他 的 任务 可 能 会 在 WaitPerson 被 唤醒 时 ， 会 突 
然 插足 并 拿 走 订 单 , 唯一 安全 的 方式 是 使 用 下 面 这 种 wait0 的 惯用 法 (当然 要 在 恰当 的 同步 内 部 ， 
并 采用 防止 错失 信号 可 能 性 的 程序 设计 ): 


while(conditionIsNotMet) 
wait; 


这 可 以 保证 在 你 退出 等 待 循环 之 前 ， 条 件 将 得 到 满足 ， 并 且 如 果 你 收 到 了 关于 某 事物 的 通 
知 ， 而 它 与 这 个 条 件 并 无 关系 (就 象 在 使 用 notifyANHO 时 可 能 发 生 的 情况 一 样 ) ， 或 者 在 你 完全 
退出 等 待 循环 之 前 ， 这 个 条 件 发 生 了 变化 ， 都 可 以 确保 你 可 以 重 返 等 待 状态 。 

请 注意 观察 ， 对 notifyAlI0 的 调用 必须 首先 捕获 waitPerson 上 的 锁 ， 而 在 WaitPerson.run0 中 
的 对 wait0 的 调用 会 自动 地 释放 这 个 锁 ， 因 此 这 是 有 可 能 实现 的 。 因 为 调用 notifyAll0 必 然 拥 有 
这 个 锁 ， 所 以 这 可 以 保证 两 个 试图 在 同一 个 对 象 上 调用 motifyA1O 的 任务 不 会 互相 冲突 。 

通过 把 整个 run0 方 法 体 放 到 一 个 try 语 句 块 中 ， 可 使 得 这 两 个 ran0 方 法 都 被 设计 为 可 以 有 序 
地 关闭 。catch 子 句 将 紧 挨 着 ran() 方 法 的 结束 括号 之 前 结束 ， 因 此 ， 如 果 这 个 任务 收 到 了 
JInterruptedException 异 常 ， 它 将 在 捕获 异常 之 后 立即 结束 。 

注意 ， 在 Chef 中 ， 在 调用 shutdownNow0 之 后 ， 你 应 该 直接 从 run0 返 回 ， 并 且 通 常 这 就 是 
你 应 该 做 的 。 但 是 ， 以 这 种 方式 执行 还 有 一 些 更 有 趣 的 东西 。 记 住 ，shutdownNow0 将 向 所 有 
由 ExecutorService 启 动 的 任务 发 送 interrupt0， 但 是 在 Chef 中 ,任务 并 没有 在 获得 该 interrupt0 
之 后 立即 关闭 ， 因 为 当 任务 试图 进入 一 个 (可 中 断 的 ) 阻塞 操作 时 ， 这 个 中 断 只 能 抛 出 
InterruptedException。 因 此 ， 你 将 看 到 首先 显示 了 “Order up!"， 然 后 当 Chef 试 图 调用 sleep0 时 ， 
抛 出 了 InterruptedException。 如 果 移 除 对 sleep0 的 调用 ， 那 么 这 个 任务 将 回 到 run0 循 环 的 顶部 ， 
并 由 于 Thread.interrupted0 测 试 而 退出 ， 同 时 并 不 抛 出 异常 。 

在 前 面 的 示例 中 ， 对 于 一 个 任务 而 言 ， 只 有 一 个 单一 的 地 点 用 于 存放 对 象 ， 从 而 使 得 另 一 
个 任务 稍 后 可 以 使 用 这 个 对 象 。 但 是 ， 在 典型 的 生产 者 -消费 者 实现 中 ， 应 使 用 先进 先 出 队列 来 
存储 被 生产 和 消费 的 对 象 。 你 将 在 本 章 稍 后 学 习 有 关 这 种 队列 的 知识 。 

练习 24: (1) 使 用 wait0 和 notifyAlI0 解 决 单个 生产 者 、 单 个 消费 者 问题 。 生 产 者 不 能 溢出 接 
收 者 的 缓冲 区 ， 而 这 在 生产 者 比 消费 者 速度 快 时 完全 有 可 能 发 生 。 如 果 消 费 者 比 生产 者 速度 快 ， 
那么 消费 者 不 能 读 取 多 次 相同 数据 。 不 要 对 生产 者 和 消费 者 的 相对 速度 作 任 何 假设 。 

练习 25，(1) 在 Restaurant,java 的 Chef 类 中 ， 在 调用 shutdownNow0 之 后 从 run0 中 return， 
观察 行为 上 的 差异 。 

练习 26，(8) 向 Restaurant.java 中 添加 一 个 BusBoy 类 。 在 上 菜 之 后 ，WaitPerson 应 该 通知 
BusBoy 清 理 。 

使 用 显 式 的 Lock 和 Condition 对 象 

{E Java SE5 的 java.util.concurrent 类 库 中 还 有 额外 的 显 式 工具 可 以 用 来 重 写 
WaxOMatic.java。 使 用 互 斥 并 允许 任务 挂 起 的 基本 类 是 Condition， 你 可 以 通过 在 Condition 上 
调用 await0 来 挂 起 一 个 任务 。 当 外 部 条 件 发 生变 化 ， 意 味 着 某 个 任务 应 该 继续 执行 时 ， 你 可 以 
通过 调用 signal0 来 通知 这 个 任务 ， 从 而 唤醒 一 个 任务 ， 或 者 调用 signalAll0 来 唤醒 所 有 在 这 个 
Condition 上 被 其 自身 挂 起 的 任务 (与 使 用 notifyAll0 相 比 ，signalAli0 是 更 安全 的 方式 )。 

下 面 是 WaxOMatic.java 的 重 写 版 本 ， 它 包含 一 个 Condition， 用 来 在 waitForWaxing() 或 
waitForBuffering0 内 部 挂 起 一 个 任务 : 


/1/: concurrency/waxomat ic2/WaxOMatic2. java 
71 Using Lock and Condition objects. 
package concurrency.waxomatic2; 
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import java.util.concurrent. 





import java.util.concurrent.locks.*; 
import static net.mindview.util.Print.*; 


class Car { 


private Lock lock = new ReentrantLock(): 
private Condition condition = lock.newCondition(); 
private boolean waxOn = false; 
public void waxed() { 
lock. Lock () 
try { 
waxOn = true; // Ready to buff 
condition.signalAll(); 
ï finally { 
lock unlock () ; 
} 
} 
public void buffed() { 
lock. tock(); “ 
try { 
waxOn = false; // Ready for another coat of wax 
condition. signalAll() ; 
} finally { 
tock.untock() ; 
» 
} 


public void waitForWaxing() throws InterruptedException { 


lock. Lock (); 

try { 
while(waxOn == false) 

condition. await(); 

} finally { 
tock.untock() : 

} 

} 


public void waitForBuffing() throws InterruptedException{ 


Lock. Lock () ; 
try { 
while(waxOn == true) 
condi tion.await(); 
} finally { 
lock.unlock(); 
} 
} 
) 


class WaxOn implements Runnable { 


private Car car; 
public WaxOn(Car c) { car = < 
public void run() { 
try { 
while(! Thread. interrupted()) { 
printnb("Wax On! "); i 
Timeunit. MILLISECONDS. sleep (208) ; 
car .waxed() ; 
car.waitForBuffing(): 





} 
} catch(InterruptedException e) { 
print("Exiting via interrupt"); 
print("Ending Wax On task"); 
} 
} 


class WaxOff implements Runnable { 


private Car car; 


public WaxOff(Car c) { car } 
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public void run() { 

try ¢ 
while(!Thread.interrupted()) { 
car.waitForWaxing(); 
printnb("Wax Off! "); 
TimeUnit .MILLISECONDS.sleep(266) ; 
car.buffed(); 


} 
} catch(InterruptedException e) { 
print("Exiting via interrupt"); 


} 
print("Ending Wax Off task"); 
} 
} 


public class WaxOMatic2 { 
public static void main(String{] args) throws Exception { 

Car car = new Car(); 
ExecutorService exec = Executors.newCachedThreadPool () ; 
exec. execute (new WaxOff (car)); 
exec. execute (new WaxOn(car)); 
TimeUnit SECONDS. sleep(5); 
exec. shutdownNow() ; 


$ 
} /* Output: (90% match) 
Wax On! Wax Off! Wax On! Wax Off! Wax On! Wax Off! Wax On! 
Wax Off! Wax On! Wax Off! Wax On! Wax Off! Wax On! Wax Off! 
Wax On! Wax Off! Wax On! Wax Off! Wax On! Wax Off! Wax On! 
Wax Off! Wax On! Wax Off! Wax On! Exiting via interrupt 
Ending Wax Off task 
Exiting via interrupt 
Ending Wax On task 
“~ 


在 Car 的 构造 器 中 ， 单 个 的 Lock 将 产生 一 个 Condition 对 象 ， 这 个 对 象 被 用 来 管理 任务 间 的 
通信 。 但 是 ， 这 个 Condition 对 象 不 包含 任何 有 关 处 理 状 态 的 信息 ， 因 此 你 需要 管理 额外 的 表示 
处 理 状 态 的 信息 ， 即 boolean waxOn.。 

每 个 对 lock0 的 调用 都 必须 紧 跟 一 个 try-finally 子 句 ， 用 来 保证 在 所 有 情况 下 都 可 以 释放 锁 。 
在 使 用 内 建 版 本 时 ， 任 务 在 可 以 调用 awaitO、signal0 或 signalAli0 之 前 ， 必 须 拥 有 这 个 锁 。 

注意 ， 这 个 解决 方案 比 前 一 个 更 加 复杂 ， 在 本 例 中 这 种 复杂 性 并 未 使 你 收获 更 多 。Lock 和 
Condition 对 象 只 有 在 更 加 困难 的 多 线程 问题 中 才 是 必需 的 。 

练习 27: (2) 修改 Restaurantjava， 使 其 使 用 显 式 的 Lock 和 Condition 对 象 。 

21.5.4 生产 者 -消费 者 与 队列 

wait0 和 notifyAlIO 方 法 以 一 种 非常 低级 的 方式 解决 了 任务 互 操作 问题 ， 即 每 次 交互 时 都 握 
手 。 在 许多 情况 下 ， 你 可 以 瞄 向 更 高 的 抽象 级 别 ， 使 用 同步 队列 来 解决 任务 协作 问题 同步 队 
列 在 任何 时 刻 都 只 允许 一 个 任务 插入 或 移 除 元 素 。 在 java.util.concurrent.BlockingQueue 接 口中 
提供 了 这 个 队列 ， 这 个 接口 有 大 量 的 标准 实现 。 你 通常 可 以 使 用 LinkedBlockingQueue， 它 是 一 
个 无 届 队 列 ， 还 可 以 使 用 ArrayBlockingQueue， 它 具有 固定 的 尺寸 ， 因 此 你 可 以 在 它 被 阻塞 之 
前 ， 向 其 中 放置 有 限 数量 的 元 素 。 

如 果 消 费 者 任务 试图 从 队列 中 获取 对 象 ， 而 该 队列 此 时 为 空 ， 那 么 这 些 队列 还 可 以 挂 起 消 
费 者 任务 ， 并 且 当 有 更 多 的 元 素 可 用 时 恢复 消费 者 任务 。 阻 塞 队列 可 以 解决 非常 大 量 的 问题 ， 
而 其 方式 与 wait0 和 notifyAIO 相 比 ， 则 简单 并 可 靠 得 多 。 

下 面 是 一 个 简单 的 测试 ， 它 将 多 个 LiftOff 对 象 的 执行 串 行 化 了 。 消 费 者 是 LiftOffRunner， 
它 将 每 个 LiftOff 对 象 从 BlockingQueue 中 推出 并 直接 运行 。( 即 ， 它 通过 显 式 地 调用 run0 而 使 用 
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自己 的 线程 来 运行 ， 而 不 是 为 每 个 任务 启动 一 个 新 线程 。) 


//: concurrency/TestBlockingQueues. java 
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import java.util.concurrent.*; 
import java.io.* 
import static net.mindview.util.Print 








class LiftOffRunner implements Runnable { 
private BlockingQueue<LiftOff> rockets; 
public LiftOffRunner(BlockingQueue<LiftOff> queue) { 
rockets = queue: 


} 
public void add(Liftoff lo) { 
try { 
rockets .put (lo); 
} catch(InterruptedException e) { 
print("Interrupted during put()"); 
} 
} 
public void run() { 
try { 
while(!Thread. interrupted()) { 
Liftoff rocket = rockets.take(); 
rocket.run(); // Use this thread 
} 
} catch(InterruptedException e) { 
print("Waking from take()"); 


} 
print("Exiting LiftOffRunner"); 
} 
} 


public class TestBlockingQueues { 
static void getkey() { 
try { 
// Compensate for Windows/Linux difference in the 
// length of the result produced by the Enter key: 
new BufferedReader ( 
new InputStreamReader (System. in)).readLine(); 
} catch(java. io. IOException e) { 
throw new RuntimeException(e); 
} 
4 
Static void getkey(String message) ( 
print (message) ; 
getkey(); 
1216 } 
static void 
test(String msg, BlockingQueue<Liftoff> queue) { 
print(msg); 
LiftoffRunner runner = new Li ftOffRunner (queue) ; 
Thread t = new Thread(runner) ; 
t.start(); 
for(int 1 = 0; 1 < 5; i++) 
runner .add(new Liftoff (5) 
getkey("Press ‘Enter’ (" + 
t.interrupt(); 
Print("Finished ”+ msg + " test"); 
} 
public static void main(String[] args) { 
test ("LinkedBlockingQueue", // Unlimited size 
new LinkedBlock ingQueue<Liftoft>()); 
test("ArrayBlockingQueue", // Fixed size 
new ArrayBlockingQueue<LiftOff>(3)); 
test("SynchronousQueue", // Size of 1 
new SynchronousQueue<Liftoff>()); 











a EDS 





玉 发 715 





} 

} 1:~ 

各 个 任务 由 main0 放 置 到 了 BlockingQueue 中 ， 并 且 由 LiftOffRunner 从 BlockingQueue 中 取 
出 。 注 意 ，LiftOffRunner 可 以 忽略 同步 问题 ， 因 为 它们 已 经 由 BlockingQueue 解 决 了 。 

练习 28，(3) 修改 TestBlockingQueue.java， 添 加 一 个 将 LiftO 人 ff 放置 到 BlockingQueue 中 的 任 
务 ， 而 不 要 放置 在 main0 中 。 

吐 司 BlockingQueue 

考虑 下 面 这 个 使 用 BlockingQueue 的 示例 ， 有 一 台 机 器 具有 三 个 任务 : 一 个 制作 吐 司 、 一 个 
给 吐 司 抹 黄油 ， 另 一 个 在 抹 过 黄油 的 吐 司 上 涂 果 获 。 我 们 可 以 通过 各 个 处 理 过 程 之 间 的 
BlockingQueue 来 运行 这 个 叶 司 制作 程序 : 


//: concurrency/ToastOMatic. java 
// A toaster that uses queues. 
import java.util.concurrent.*; 
import java.util 
import static net.mindview.util.Print.*; 








class Toast { 
public enum Status { DRY, BUTTERED, JAMMED } 
private Status status = Status.DRY; 
private final int id; 
public Toast(int idn) { id = idn; } 
public void butter() { status = Status.BUTTERED; } 
public void jam() { status = Status.JAMMED; } 
public Status getStatus() { return status; } 
public int getId() { return id; } 
public String toString() { 
return “Toast "+ id + ": “ + status; 
} 
} 





class ToastQueue extends LinkedBlockingQueue<Toast> {} 


class Toaster implements Runnable { 
private ToastQueue toastQueue; 
private int count = 6; 
private Random rand = new Random(47) ; 
public Toaster(ToastQueue tq) { toastQueue = tq; } 
public void run() { 
try ( 
while(!Thread. interrupted()) { 
TimeUnit MILLISECONDS. sleep( 
100 + rand.nextInt (560); 
// Make toast 
Toast t = new Toast(count++); 
print(t): 
// Insert into queue 
toastQueue. put (t); 


4 
} catch(InterruptedException e) { 
print("Toaster interrupted"); 


} 
print("Toaster off"); 


} 


// Apply -butter to toast: 
class Butterer implements Runnable { 
private ToastQueue dryQueue, butteredQueue; 
public Butterer(ToastQueue dry, ToastQueue buttered) { 
dryQueue = dry; 
butteredQueue = buttered: 
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} 
public void run() { 
try { 
while(!Thread.interrupted()) { 
// Blocks until next piece of toast is available: 
Toast t = dryQueue.take(); 
t.butter(); 
print(t); 
butteredQueue. put (t); 


} 
} catch(InterruptedException e) { 
print("Butterer interrupted"); 
F 
print("Butterer off"); 
} 
} 


// Apply jam to buttered toast: 
class Jammer implements Runnable { 
private ToastQueue butteredQueue, finishedQueue; 
public Jammer (ToastQueue buttered, ToastQueue finished) { 
butteredQueue = buttered; 
finishedQueue = finished; 
} 
public void run() { 
try { 
while(! Thread. interrupted()) { 
// Blocks until next piece of toast is available: 
Toast t = butteredQueve. take(); 
t.jamQ); 
print(t); 
finishedQueue.put(t); 
} 
} catch(InterruptedException e) { 
print("Jammer interrupted"); 
} 
print("Janmer off"); 
} 
} 


// Consume the toast: 
class Eater implements Runnable { 
private ToastQueue finishedQueue; 
private int counter = 0; 
Public Eater (ToastQueue finished) { 
finishedQueue = finished; 
} 
public void run() { 
try { 
while(!Thread.interrupted()) { 
// Blocks until next piece of toast is available: 
Toast t = finishedQueue.take(); 
// Verify that the toast is coming in order, 
// and that all pieces are getting jammed: 
if(t.getId() != counter++ || 
t.getStatus() != Toast.Status. JAMMED) { 
print(">>>> Error: * + t); 
System.exit(1); 
} else 
print("Chomp! " + t); 
} 


} catch(InterruptedException e) { 
print("Eater interrupted"); 
+ 
print("Eater off"); 
} 





} 
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public class ToastOMatic { 
public static void main(String(] args) throws Exception { 
ToastQueue dryQueue = new ToastQueue(), 
butteredQueue = new ToastQueue(), 
finishedQueue = new ToastQueue(); 
ExecutorService exec = Executors .newCachedThreadPool () ; 
exec.execute(new Toaster (dryQueue)) ; 
exec.execute(new Butterer(dryQueue, butteredQueue)) ; 
exec.execute(new Jammer (butteredQueue, finishedQueue)); 
exec.execute(new Eater(finishedQueue)); 
TimeUnit. SECONDS. sleep(5); 
exec. shutdownNow() ; 


} 
} /* (Execute to see output) *///:~ 


Toast 是 一 个 使 用 enum 值 的 优秀 示例 。 注 意 ， 这 个 示例 中 没有 任何 显 式 的 同步 (即使 用 Lock 
对 象 或 synehronized 关 键 字 的 同步 )， 因 为 同步 由 队列 (其 内 部 是 同步 的 ) 和 系统 的 设计 隐 式 地 
管理 了 一 一 每 片 Toast 在 任何 时 刻 都 只 由 一 个 任务 在 操作 。 因 为 队列 的 阻塞 使 得 处 理 过 程 将 被 
自动 地 挂 起 和 恢复 。 你 可 以 看 到 由 BlockingQueue 产 生 的 简化 十 分 明显 。 在 使 用 显 式 的 wait0 和 
motifyAUO 时 存在 的 类 和 类 之 间 的 耦合 被 消除 了 ， 因 为 每 个 类 都 只 和 它 的 BlockingQueue 通 信 。 

练习 29: (8) 修改 ToastOMaticjava， 使 用 两 个 单独 的 组 装 线 来 创建 徐 有 花生 黄油 和 果冻 的 
吐 司 三 明治 (一 个 用 于 花生 黄油 ， 第 二 个 用 于 果冻 ， 然 后 把 两 条 线 合并 )。 


21.5.5 任务 间 使 用 管道 进行 输入 /输出 

通过 输入 /输出 在 线程 间 进 行 通信 通常 很 有 用 。 提 供 线程 功能 的 类 库 以 “管道 ”的 形式 对 线 
程 间 的 输入 /输出 提供 了 支持 。 它 们 在 Java 输 入 /输出 类 库 中 的 对 应 物 就 是 PipedWriter 类 (允许 任 
务 向 管道 写 ) 和 PipedReader 类 (允许 不 同 任务 从 同一 个 管道 中 读 取 )。 这 个 模型 可 以 看 成 是 
“生产 者 一 消费 者 ”问题 的 变 体 ， 这 里 的 管道 就 是 一 个 封装 好 的 解决 方案 。 管 道 基本 上 是 一 个 阻 
塞 队列 ， 存 在 于 多 个 引入 BlockingQueue 之 前 的 Java 版 本 中 。 

下 面 是 一 个 简单 例子 ， 两 个 任务 使 用 一 个 管道 进行 通信 : 

/1: concurrency/Piped1o. java 

// Using pipes for inter-task I/0 

import java.util.concurrent.*; 

import java.io.*; 


import java.util.*; 
import static net.mindview.util.Print.*; 


class Sender implements Runnable { 
private Random rand = new Random(47); 
private PipedWriter out = new PipedWriter(); 
Public PipedWriter getPipedwWriter() { return out; } 
public void run() { 
try { 
while(true) 
for(char c = 'A'; c <= 'z'; c++) { 
out .write(c); 
TimeUnit MILLISECONDS. sleep(rand.nextInt (588) ); 


} 
catch(I0Exception e) { 
print(e + " Sender write exception"); 
catch(InterruptedException e) { 
print(e + " Sender sleep interrupted"): 


} 


class Receiver implements Runnable { 
private PipedReader in: 
public Receiver(Sender sender) throws IOException { 
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in = new PipedReader (sender .getPipedwriter()); 


} 
public void run() { 
try { 
while(true) { 
// Blocks until characters are there: 
printnb("Read: " + (char)in.read() +", "); 


} 
} catch(IOException e) { 
print(e + " Receiver read exception"); 
y 
$ 
} 


public class PipedIO { 
public static void main(String[] args) throws Exception { 

Sender sender = new Sender(); 
Receiver receiver = new Receiver (sender) ; 
ExecutorService exec = Executors.newCachedThreadPool () ; 
exec. execute (sender) ; 
exec. execute(receiver) ; 
TimeUnit. SECONDS. sleep(4) ; 
exec. shutdownNow () ; 


} 
} /* Output: (65% match) 
Read: A, Read: B, Read: C, Read: D, Read: E, Read: F, Read: 
G, Read: H, Read: I, Read: J, Read: K, Read: L, Read: M, 
java. lang. InterruptedException: sleep interrupted Sender 
sleep interrupted 
java.io. InterruptedIOException Receiver read exception 
Whim 


Sender 和 Receiver 代 表 了 需要 互相 通信 两 个 任务 。Sender 创 建 了 一 个 PipedWriter， 它 是 一 
个 单独 的 对 象 ， 但 是 对 于 Receiver，PipedReader 的 建立 必须 在 构造 器 中 与 一 个 PipedWriter 相 关 
联 。Sender 把 数据 放 进 Writer， 然 后 休眠 一 段 时 间 (随机 数 ) 。 然 而 ，Receiver 没 有 sleep0 和 
wait0。 但 当 它 调用 read0 时 ， 如 果 没 有 更 多 的 数据 ， 管 道 将 自动 阻塞 。 

注意 sender 和 receiver 是 在 main0 中 启动 的 ， 即 对 象 构造 彻底 完毕 以 后 。 如 果 你 启动 了 一 个 
没有 构造 完毕 的 对 象 ， 在 不 同 的 平台 上 管道 可 能 会 产生 不 一 致 的 行为 注意，BlockingQueue 使 
用 起 来 更 加 健壮 而 容易 ) 。 

在 shutdownNow0 被 调用 时 ， 可 以 看 到 PipedReader 与 普通 IO 之 间 最 重要 的 差异 一 Piped- 
Reader 是 可 中 断 的 。 如 果 你 将 in.readO 调 用 修改 为 System.in.read0， 那 么 interruptO 将 不 能 打 断 
read0 调 用 。 

练习 30: (1) 修改 PipedIOjava， 使 其 使 用 BlockingQueue 而 不 是 管道 。 


21.6 死 锁 


现在 你 理解 了 ， 一 个 对 象 可 以 有 synchronized 方 法 或 其 他 形式 的 加 锁 机 制 来 防止 别 的 任务 在 
互 斥 还 没有 释放 的 时 候 就 访问 这 个 对 象 。 你 已 经 学 习 过 ， 任 务 可 以 变 成 阻塞 状态 ， 所 以 就 可 能 
出 现 这 种 情况 : 某 个 任务 在 等 待 另 一 个 任务 ， 而 后 者 又 等 待 别 的 任务 ， 这 样 一 直下 去 ， 直 到 这 
个 链条 上 的 任务 又 在 等 待 第 一 个 任务 释放 锁 。 这 得 到 了 一 个 任务 之 间 相互 等 待 的 连续 循环 ， 没 
有 哪个 线程 能 继续 。 这 被 称 之 为 死 锁 。 。 

如 果 你 运行 一 个 程序 ， 而 它 马上 就 死 锁 了 ， 你 可 以 立即 跟踪 下 去 。 真 正 的 问题 在 于 ， 程 序 


可 能 看 起 来 工作 良好 ， 但 是 具有 潜在 的 死 锁 危险 。 这 时 ， 死 锁 可 能 发 生 ， 而 事先 却 没有 任何 


日 。 当 两 个 任务 可 以 修改 它们 的 状态 (它们 不 会 阻塞 ) 时 ， 你 还 可 以 使 用 活 锁 ,但 是 这 么 做 不 会 得 到 什么 有 用 的 改进 。 
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征兆 ， 所 以 缺陷 会 潜伏 在 你 的 程序 里 ， 直 到 客户 发 现 它 出 乎 意料 地 发 生 〈 以 一 种 几乎 肯定 是 
很 难 重 现 的 方式 发 生 ) 。 因 此 ， 在 编写 并 发 程序 的 时 候 ， 进 行 仔细 的 程序 设计 以 防止 死 锁 是 关 
键 部 分 。 

由 Edsger Dijkstra 提 出 的 藻 学 家 就 艾 问 题 是 一 个 经 典 的 死 锁 例证 。 该 问题 的 基本 描述 中 是 指 
定 五 个 哲学 家 (不 过 这 里 的 例子 中 将 允许 任意 数目 )。 这 些 哲 学 家 将 花 部 分 时 间 思 考 ， 花 部 分 时 
间 就 餐 。 当 他 们 思考 的 时 候 ， 不 需要 任何 共享 资源 ， 但 当 他 们 就 餐 时 ， 将 使 用 有 限 数量 的 餐具 。 
在 问题 的 原始 描述 中 ， 餐 具 是 又 子 。 要 吃 到 桌子 中 央 盘 子 里 的 意大利 面条 需要 用 两 把 又 子 ， 不 
过 把 餐具 看 成 是 竹子 更 合理 ， 很 明显 ， 哲 学 家 要 就 餐 就 需要 两 根 秘 子 。 

问题 中 引入 的 难点 是 : 作为 哲学 家 ， 他 们 很 穷 ， 所 以 他 们 只 能 买 五 根 筷子 (更 一 般 地 讲 ， 
筷子 和 哲学 家 的 数量 相同 ) 。 他 们 围 坐 在 桌子 周围 ， 每 人 之 间 放 一 根 筷子 。 当 一 个 哲学 家 要 就 餐 
的 时 候 ， 这 个 哲学 家 必须 同时 得 到 左边 和 右边 的 筷子 。 如 果 一 个 哲学 家 左边 或 右边 已 经 有 人 在 
使 用 筷子 了 ， 那 么 这 个 后 学 家 就 必须 等 待 ， 直 至 可 得 到 必需 的 筷子 。 


//: concurrency/Chopstick. java 
// Chopsticks for dining philosophers. 


public class Chopstick { 
private boolean taken = false; 
public synchronized 
void take() throws InterruptedException { 
while (taken) 
wait; 
taken = true; 
} 
public synchronized void drop() { 
taken = false; 
notifyALl(); 


YM hx 


任何 两 个 Philosopher 都 不 能 成 功 take0 同 一 根 饶 子 。 另 外 ， 如 果 一 根 Chopstick 已 经 被 某 个 
Philosopher 获 得 ， 那 么 另 一 个 Philosopher 可 以 waitO ， 直 至 这 根 Chopstick 的 当前 持 有 者 调用 
drop0 使 其 可 用 为 止 。 


当 一 个 Philosopher 任 务 调用 take0 时 ， 这 个 Philosopher 将 等 待 ， 直 至 taken 标 志 变 为 false (Hi 
至 当前 持 有 Chopstick 的 Philosopher 释 放 它 ) 。 然 后 这 个 任务 会 将 taken 标 志 设置 为 true， 以 表示 
现在 由 新 的 Philosopher 持 有 这 根 Chopstick。 当 这 个 Philosopher 使 用 完 这 根 Chopstick 时 ， 它 会 调 
用 drop0 来 修改 标志 的 状态 ， 并 notifyAll0 所 有 其 他 的 Philosopher， 这 些 Philosopher 中 有 些 可 能 
就 在 wait0 这 根 Chopstick。 


//: concurrency/Philosopher .java 

// A dining philosopher 

import java.util.concurrent.*; 

import java.util.*; 

import static net.mindview.util.Print.*; 


public class Philosopher implements Runnable { 

private Chopstick left; 

private Chopstick right; 

private final int id: 

private final int ponderFactor; 

private Random rand = new Random(47); 

Private void pause() throws InterruptedException { 
if(ponderFactor == @) return; 
TimeUnit MILLISECONDS. sleep( 

rand.nextInt(ponderFactor * 250)); 
} 
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public Philosopher (Chopstick left, Chopstick right, 
int ident, int ponder) { 
this.left = left; 
this.right = right; 





id = ident; 
ponderFactor = ponder: 
} 
public void run() { 
try { 


while(!Thread.interrupted()) { 
print(this + * " + “thinking"); 
pause(); 
// Philosopher becomes hungry 
print(this +" ”+ "grabbing right"); 
right. take(); 
print(this + " " + "grabbing left"); 
left.take(); 
print(this +" " + "eating"); 
pause(); 
right.drop(); 
left.drop(); 
} 
} catch(InterruptedException e) { 
print(this + " " + “exiting via interrupt"); 
} 
} 
public String toString() { return "Philosopher "+ id; } 
y M~ 


在 Philosopher, run0 中 ， 每 个 Philosopher 只 是 不 断 地 思考 和 吃饭 。 如 果 PonderFactor 不 为 0， 


则 pause( 方 法 会 休眠 (sleeps) 一 段 随机 的 时 间 。 通 过 使 用 这 种 方式 ， 你 将 看 到 Philosopher 会 
在 思考 上 花 掉 一 段 随机 化 的 时 间 ， 然 后 尝试 着 获取 (take0) 右边 和 左边 的 Chopstick， 随 后 在 吃 
饭 上 再 花 掉 一 段 随机 化 的 时 间 ， 之 后 重复 此 过 程 。 


现在 我 们 可 以 建立 这 个 程序 的 将 会 产生 死 锁 的 版 本 了 : 


1/: concurrency/DeadtockingDiningPhilosophers. java 
(/ Demonstrates how deadlock can be hidden in a program. 
// {Args: @ 5 timeout} 

import java.util.concurrent.*; 


public ctass DeadlockingDiningPhilosophers { 
Public static void main(String{] args) throws Exception { 
int ponder = 5; 
if(args.length > 0) 
ponder = Integer .parseInt(args[0]); 
int size = 5; 
if(args.length > 1) 
size = Integer.parseInt(args(1]); 
ExecutorService exec = Executors.newCachedThreadPool () ; 
Chopstick[] sticks = new Chopstick[size]; 
for(int 1 = @; 1 < size; i++) 
sticks[1] = new Chopstick(); 
for(int 1 = 0; i < size: i++) 
exec. execute(new Philosopher ( 
sticks[i], sticks[(i+1) % size], i, ponder)); 
if (args.length 3 && args[2].equals("timeout")) 
TimeUnit. SECONDS. sleep(5) ; 
else { 
System.out.printin("Press ‘Enter’ to quit"); 
System. in.read(); 





} 
exec. shutdownNow() ; 


} 
} /* (Execute to see output) *///:~ 
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你 会 发 现 ， 如 果 Philosopher 花 在 思考 上 的 时 间 非 常 少 ， 那 么 当 他 们 想 要 进餐 时 ， 全 都 会 在 
Chopstick 上 产生 竞争 ， 而 死 锁 也 就 会 更 快 地 发 生 。 

第 一 个 命令 行 参数 可 以 调整 ponder 因 子 ， 从 而 影响 每 个 Philosopher 花 费 在 思考 上 的 时 间 长 
度 。 如 果 有 许多 Philosopher， 或 者 他 们 花费 很 多 时 间 去 思考 ， 那 么 尽管 存在 死 锁 的 可 能 ， 但 你 
可 能 永远 也 看 不 到 死 锁 。 值 为 0 的 命令 行 参数 倾向 于 使 死 锁 尽快 发 生 。 

注意 ，Chopstick 对 象 不 需要 内 部 标识 符 ， 它 们 是 由 在 数组 sticks 中 的 位 置 来 标识 的 。 每 个 
了 Philosopher 构 造 器 都 会 得 到 一 个 对 左边 和 右边 Chopstick 对 象 的 引用 。 除 了 最 后 一 个 Philosopher， 
其 他 所 有 的 Philosopher 都 是 通过 将 这 个 Philosopher 定 位 于 下 一 对 Chopstick 对 象 之 间 而 被 初始 化 
的 ， 而 最 后 一 个 Philosopher 右 边 的 Chopstick 是 第 0 个 Chopstick， 这 样 这 个 循环 表 也 就 结束 了 。 
因为 最 后 一 个 Philosopher 坐 在 第 一 个 Philosopher 的 右边 ， 所 以 他 们 会 共享 第 0 个 Chopstick。 现 
在 ， 所 有 的 Philosopher 都 有 可 能 希望 进餐 ， 从 而 等 待 其 临近 的 Philosopher 放 下 它们 的 Chopstick。 
这 将 使 程序 死 锁 。 

如 果 Philosopher 花 费 更 多 的 时 间 去 思考 而 不 是 进餐 (使 用 非 0 的 ponder 值 ， 或 者 大 量 的 
Philosopher) ， 那 么 他 们 请 求 共享 资源 (Chopstick) 的 可 能 性 就 会 小 许多 ， 这 样 你 就 会 确信 该 程 
序 不 会 死 锁 ， 尽 管 它们 并 非 如 此 。 这 个 示例 相当 有 趣 ， 因 为 它 演示 了 看 起 来 可 以 正确 运行 ， 但 
实际 上 会 死 锁 的 程序 。 

要 修正 死 锁 问 题 ， 你 必须 明白 ， 当 以 下 四 个 条 件 同 时 满足 时 ， 就 会 发 生死 锁 ， 

D 互 斥 条 件 。 任 务 使 用 的 资源 中 至 少 有 一 个 是 不 能 共享 的 。 这 里 ， 一 根 Chopstick 一 次 就 只 
能 被 一 个 Philosopher 使 用 。 

2) 至 少 有 一 个 任务 它 必须 持 有 一 个 资源 且 正在 等 待 获 取 一 个 当前 被 别 的 任务 持 有 的 资源 。 
也 就 是 说 ， 要 发 生死 锁 ，Philosopher 必 须 拿 着 一 根 Chopstick 并 且 等待 另 一 根 。 

3) 资源 不 能 被 任务 抢占 ， 任 务必 须 把 资源 释放 当 作 普通 事件 。Philosopher 很 有 礼 狐 ， 他 们 
不 会 从 其 他 Philosopher 那 里 抢 Chopstick。 

4) 必须 有 循环 等 待 ， 这 时 ， 一 个 任务 等 待 其 他 任务 所 持 有 的 资源 ， 后 者 又 在 等 待 另 一 个 任 
务 所 持 有 的 资源 ， 这 样 一 直下 去 ， 直 到 有 一 个 任务 在 等 待 第 一 个 任务 所 持 有 的 资源 ， 使 得 大 家 
都 被 锁 住 。 在 DeadlockingDiningPhilosophersjava 中 ， 因 为 每 个 Philosopher 都 试图 先 得 到 右边 的 
Chopstick， 然 后 得 到 左边 的 Chopstick， 所 以 发 生 了 循环 等 待 。 

因为 要 发 生死 锁 的 话 ， 所 有 这 些 条 件 必须 全 部 满足 ， 所 以 要 防止 死 锁 的 话 ， 只 需 破坏 其 中 
一 个 即 可 。 在 程序 中 ， 防 止 死 锁 最 容易 的 方法 是 破坏 第 4 个 条 件 。 有 这 个 条 件 的 原因 是 每 个 
了 Philosopher 都 试图 用 特定 的 顺序 拿 Chopstick: 先 右 后 左 。 正 因为 如 此 ， 就 可 能 会 发 生 “ 每 个 人 
都 拿 着 右边 的 Chopstick， 并 等 待 左边 的 Chopstick” 的 情况 ， 这 就 是 循环 等 待 条 件 。 然 而 ， 如 果 
最 后 一 个 Philosopher 被 初始 化 成 先 拿 左边 的 Chopstick， 后 拿 右边 的 Chopstick， 那 么 这 个 
Philosopher 将 永远 不 会 阻止 其 右边 的 Philosopher 拿 起 他 们 的 Chopstick。 在 本 例 中 ， 这 就 可 以 防 
止 循环 等 待 。 这 只 是 问题 的 解决 方法 之 一 ， 也 可 以 通过 破坏 其 他 条 件 来 防止 死 锁 (具体 细节 请 
参考 更 高 级 的 讨论 线程 的 书籍 ) : 


//;_concurrency/FixedDiningPhilosophers. java 
// Dining philosophers without deadlock. 

// {Args: 5 5 timeout} 

import java.util.concurrent.*; 





public class FixedDiningPhilosophers { 
Public static void main(String{] args) throws Exception { 
int ponder = 5; 
if (args. length > @) 
ponder = Integer.parseInt(args[@]); 
int size = 5; 
if(args.length > 1) 
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size = Integer .parseInt(args[1]); 
ExecutorService exec = Executors.newCachedThreadPool(); 
Chopstick[] sticks = new Chopstick[size] ; 
for(int i = 0; 1 < size; i++) 
sticks[i] = new Chopstick(); 
for(int i =; 1 < size; i++) 
if(i < (size-1)) 
exec.execute(new Philosopher( 
Sticks{i], sticks[i+1], i, ponder)); 
else 
exec.execute(new Philosopher( 
sticks[0], sticks[i], i, ponder)); 
if(args.length == 3 && args[2] .equals("timeout")) 
TimeUnit. SECONDS. steep(5) ; 
else { 
System.out.printin(“Press ‘Enter’ to quit"); 
System. in.read(); 





exec. shutdownNow() ; 
} 
} /* (Execute to see output) *///:~ 


通过 确保 最 后 一 个 Philosopher 先 拿 起 和 放下 左边 的 Chopstick， 我 们 可 以 移 除 死 锁 ， 从 而 使 
这 个 程序 平滑 地 运行 。 

Java 对 死 锁 并 没有 提供 语言 层面 上 的 支持 ， 能 否 通过 仔细 地 设计 程序 来 避免 死 锁 ， 这 取决 于 
你 自己 。 对 于 正在 试图 调试 一 个 有 死 锁 的 程序 的 程序 员 来 说 ， 这 不 是 什么 安慰 人 的 话 。 

练习 31; (8) 修改 DeadlockingDiningPhilosophersjava， 使 得 当 哲学 家 用 完 窝 子 之 后 ， 把 包 
子 放 在 一 个 筷 乱 里 。 当 哲学 家 要 就 餐 的 时 候 ， 他 们 就 从 筷 笼 里 取出 下 两 根 可 用 的 筷子 。 这 消除 
了 死 锁 的 可 能 吗 ? 你 能 仅仅 通过 减少 可 用 的 筷子 数目 就 重新 引入 死 锁 吗 ? 


21.7 新 类 库 中 的 构件 


Java SE5 的 java.util.concurrent 引 入 了 大 量 设计 用 来 解决 并 发 问题 的 新 类 。 学 习 使 用 它们 将 
有 助 于 你 编写 出 更 加 简单 而 健壮 的 并 发 程序 。 

本 节 包 含 了 各 种 组 件 具有 代表 性 的 示例 ， 但 是 少数 组 件 ， 即 那些 你 不 太 可 能 会 用 到 或 磁 到 
的 组 件 ， 没 有 包括 在 内 。 

因为 这 些 组 件 设计 各 种 问题 ， 所 以 没有 一 种 清晰 的 方式 可 以 用 来 组 织 它 们 ， 因此 我 尝试 着 
从 最 简单 的 示例 入手 ， 逐 渐 增加 复杂 度 ， 从 而 介绍 所 有 的 示例 。 

21.7.1 CountDownLatch 

它 被 用 来 同步 一 个 或 多 个 任务 ， 强 制 它们 等 待 由 其 他 任务 执行 的 一 组 操作 完成 。 

你 可 以 向 CountDownLatch 对 象 设置 一 个 初始 计数 值 ， 任何 在 这 个 对 象 上 调用 wait0 的 方法 都 
将 阻塞 ， 直 至 这 个 计数 值 到 达 0。 其 他 任务 在 结束 其 工作 时 ， 可 以 在 该 对 象 上 调用 countDown0 来 
减 小 这 个 计数 值 。CountDownLateh 被 设计 为 只 触发 一 次 ， 计 数值 不 能 被 重 置 。 如 果 你 需要 能 够 
重 置 计数 值 的 版 本 ， 则 可 以 使 用 CyclicBarrier。 

调用 countDown0 的 任务 在 产生 这 个 调用 时 并 没有 被 阻塞 ， 只 有 对 await0 的 调用 会 被 阻塞 ， 
直至 计数 值 到 达 0。 

CountDownLateh 的 典型 用 法 是 将 一 个 程序 分 为 n 个 互相 独立 的 可 解决 任务 ， 并 创建 值 为 0 的 
CountDownLatch。 当 每 个 任务 完成 时 ， 都 会 在 这 个 锁 存 器 上 调用 countDown0。 等 待 问 题 被 解 
决 的 任务 在 这 个 锁 存 器 上 调用 await0， 将 它们 自己 拦住 ， 直 至 锁 存 器 计数 结束 。 下 面 是 演示 这 
种 技术 的 一 个 框架 示例 : 


//: concurrency/CountDownLatchDemo. java 





| 
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import java.util.concurrent.*: 
import java.util.*; 
import static net.mindview.util.Print.*; 





// Performs some portion of a task: 
class TaskPortion implements Runnable { 
private static int counter = @; 
private final int id = counter++; 
private static Random rand = new Random(47) ; 
private final CountDownLatch latch; 
TaskPortion(CountDownLatch latch) { 
this.latch = latch: 
} 
public void run() { 
try { 
doWork(); 
latch. countDown() ; 
} catch(InterruptedException ex) { 
// Acceptable way to exit 
水 
} 
public void doWork() throws InterruptedException { 
TimeUnit . MILLISECONDS. sleep(rand.nextInt (2868) ) ; 
print(this + "completed"); 
} 
public String toString() { 
return String. format("%1$-3d ", id); 
¥ 
) 


// Waits on the CountDownLatch: 
class WaitingTask implements Runnable { 
private static int counter = 0; 
private final int id = counter++; 
private final CountDownLatch latch; 
WaitingTask(CountDownLatch latch) { 
this.latch = latch; 
} 
public void run() { 
try { 
latch. await(); 
print(*Latch barrier passed for " + this); 
} catch(Interruptedéxception ex) { 
print(this + * interrupted"); 
} 
} 
public String toString() { 
return String. format ("WaitingTask %1$-3d ", id); 
} 
} 


public class CountDownLatchDemo { 
static final int SIZE = 100; 
public static void main(String[] args) throws Exception { 
ExecutorService exec = Executors.newCachedThreadPool () ; 
// ALL must share a single CountDownLatch object: 
CountDownLatch latch = new CountDownLatch(SIZE) ; 
for(int 1 = 0; 1 < 10; i++) 
exec.execute(new WaitingTask(latch)) ; 
for(int í = 9; í < SIZE; i++) 
exec.execute(new TaskPortion(latch)); 
print("Launched all tasks"); 
; exec. shutdown(); // Quit when all tasks complete 
} /* (Execute to see output) *///:~ 





TaskPortion 将 随机 地 休眠 一 段 时 间 ， 以 模拟 这 部 分 工作 的 完成 ， 而 WaitingTask 表 示 系 统 中 








[1231] 
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必须 等 待 的 部 分 ， 它 要 等 待 到 问题 的 初始 部 分 完成 为 止 。 所 有 任务 都 使 用 了 在 main0 中 定义 的 同 
一 个 单一 的 CountDownLatch 。 

练习 32: (7) 使 用 CountDownLatch 解 决 OrnamentalGarden.java 中 Entrance 产 生 的 结果 互相 
关联 的 问题 。 从 新 版 本 的 示例 中 移 除 不 必要 的 代码 。 

类 库 的 线程 安全 

注意 ，TaskPortion 包 含 一 个 静态 的 Random 对 象 ， 这 意味 着 多 个 任何 可 能 会 同时 调用 
Random.nextInt0。 这 是 否 安 全 呢 ? 

如 果 存 在 问题 ， 在 这 种 情况 下 ， 可 以 通过 向 TaskPortion 提 供 其 自己 的 Random 对 象 来 解决 。 
也 就 是 说 ， 通 过 移 除 static 限 定 符 的 方式 解决 。 但 是 这 个 问题 对 于 Java 标 准 类 库 中 的 方法 来 说 ， 
也 大 都 存在 : 哪些 是 线程 安全 的 ? 哪些 不 是 ? 

遗 展 的 是 ，JDK 文 档 并 没有 指出 这 一 点 。Random.nextInt0 碰 巧 是 安全 的 ， 但 是 ， 你 必须 通 
过 使 用 Web 引 擎 ， 或 者 审视 Java 类 库 代 码 ， 去 逐个 地 揭示 这 一 点 。 这 对 于 被 设计 为 支持 ， 至 少 理 
论 上 支持 并 发 的 程序 设计 语言 来 说 ， 并 非 是 一 件 好 事 。 

21.7.2 CyclicBarrier 

CydlicBarrier 适 用 于 这 样 的 情况 ， 你 希望 创建 一 组 任务 ， 它 们 并 行 地 执行 工作 ， 然 后 在 进行 
下 一 个 步 允 之 前 等 待 ， 直 至 所 有 任务 都 完成 (看 起 来 有 些 像 join0)。 它 使 得 所 有 的 并 行 任务 都 将 
在 栅栏 处 列队 ， 因 此 可 以 一 致 地 向 前 移动 。 这 非常 像 CountDownLatch， 只 是 CountDownLatch 
是 只 触发 一 次 的 事件 ， 而 CyclicBarrier 可 以 多 次 重用 。 

从 刚 开始 接触 计算 机 时 开始 ， 我 就 对 仿真 着 了 迷 ， 而 并 发 是 使 仿真 成 为 可 能 的 一 个 关键 因 
素 。 记 得 我 最 开始 编写 的 一 个 程序 就 是 一 个 仿真 : 一 个 用 BASIC 编 写 的 〈 由 于 文件 名 的 限制 而 ) 
命名 为 HOSRAC.BAS 的 赛马 游戏 。 下 面 是 那个 程序 的 面向 对 象 的 多 线程 版 本 ， 其 中 使 用 了 
CyclicBarrier ; 


1/: concurrency/HorseRace. java 
// Using CyclicBarriers. 

import java.util.concurrent.*; 

import java.util.*; 

import static net.mindview.util.Print.*; 





class Horse implements Runnable ( 
private static int counter = 9; 
private final int id = counter++; 
private int strides = 0; 
private static Random rand = new Random(47) ; 
private static CyclicBarrier barrier; 
public Horse(CyclicBarrier b) { barrier = b; } 
public synchronized int getStrides() { return strides; } 
public void run() { 
try { 
while(!Thread.interrupted()) { 
synchronized(this) { 
strides += rand.nextInt(3); // Produces @, 1 or 2 


} 
barrier.await(); 
} 
} catch(InterruptedException e) { 
// A legitimate way to exit 


} catch(BrokenBarrierException e) { 
// This one we want to know about 


O 那 时 我 还 是 高 中 的 新 生 ， 教 室 里 有 一 台 ASR-33 电 传 打字 机 ， 通 过 波 特 率 为 110 的 声音 耦合 调制 解 调 器 来 访问 一 
HP-1000. 
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throw new RuntimeException(e); 
$ 


} 
public String toString() { return "Horse "+ id +" ": } 


public String tracks() { FE 
StringBuilder s = new StringBuilder(); 
for(int 1 =@; i < getStrides(); i++) 
s.append("*" 
s.append( id): 
return s.toString(); 
} 
} 


public class HorseRace { 
static final int FINISH_LINE = 75; 
private List<Horse> horses = new ArrayList<Horse>(); 
Private ExecutorService exec = 
Executors .newCachedThreadPool () ; 
private CyclicBarrier barrier: 
public HorseRace(int nHorses, final int pause) { 
barrier = new CyclicBarrier(nHorses, new Runnable() { 
public void run() { 
StringBuilder s = new StringBuilder (); 
for(int 1 = 6; 1 < FINISH_LINE; 1++) 
S.append("="); // The fence on the racetrack 
print(s); 
for (Horse horse : horses) 
print (horse. tracks()); 
for (Horse horse : horses) 
if(horse.getStrides() >= FINISH_LINE) { 
print(horse + "won!"); 
exec. shutdownNow() ; 
return; 
} 
try { 
TimeUnit MILLISECONDS. steep (pause); 
} catch(InterruptedException e) { 
print("barrier-action sleep interrupted"); 
} 























} 
yi 
for(int i = @; i < nHorses; i++) { 
Horse horse = new Horse(barrier); 
horses.add(horse); 
exec. execute (horse); 
} 
$: 
public static void main(String[] args) { 
int nHorses = 7; 
int pause = 200; 1234] 
if(args.length > 9) { // Optional argument 
int n = new Integer(args[@)); 
nHorses =n > @? n : nHorses: 
} 
if(args.length > 1) { // Optional argument 














int p = new Integer (args{1]); 
pause = p > -1 ? p : pause: 
d 
new HorseRace(nHorses, pause); 


} 
} /* (Execute to see output) *///:~ 


1 可 以 向 CyclicBarrier 提 供 一 个 “栅栏 动作 " ， 它 是 一 个 Runnable， 当 计数 值 到 达 0 时 自动 执 
| 行 -一 这 是 CyclicBarrier 和 CountDownLateh 之 间 的 另 一 个 区 别 。 这 里 ， 栅 栏 动作 是 作为 匿名 内 
| 部 类 创建 的 ， 它 被 提交 给 了 CyclicBarrier 的 构造 器 。 











1235) 
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我 试图 让 每 匹 马 都 打印 自己 ， 但 是 之 后 的 显示 顺序 取决 于 任务 管理 器 。CyclicBarrier 使 得 每 
匹 马 都 要 执行 为 了 向 前 移动 所 必需 执行 的 所 有 工作 ， 然 后 必须 在 栅栏 处 等 待 其 他 所 有 的 马 都 准 
备 完毕 。 当 所 有 的 马 都 向 前 移动 时 ，CyclicBarrier 将 自动 调用 Runnable 栅 栏 动作 任务 ， 按 顺序 
显示 马 和 终点 线 的 位 置 。 

一 且 所 有 的 任务 都 越过 了 栅栏 ， 它 就 会 自动 地 为 下 一 回合 比赛 做 好 准备 。 

为 了 展示 这 个 非常 简单 的 动画 效果 ， 你 需要 将 控制 台 视 窗 的 尺寸 调整 为 小 到 只 有 马 时 ， 才 
会 展示 出 来 。 
21.7.3 DelayQueue 

这 是 一 个 无 界 的 BlockingQueue， 用 于 放置 实现 了 Delayed 接 口 的 对 象 ， 其 中 的 对 象 只 能 在 其 
到 期 时 才能 从 队列 中 取 走 。 这 种 队列 是 有 序 的 ， 即 队 头 对 象 的 延迟 到 期 的 时 间 最 长 。 如 果 没有 
任何 延迟 到 期 ， 那 么 就 不 会 有 任何 头 元 素 ， 并 且 poll0 将 返回 null ( 正 因为 这 样 ， 你 不 能 将 null 放 
置 到 这 种 队列 中 ) 。 

下 面 是 一 个 示例 ， 其 中 的 Delayed 对 象 自身 就 是 任务 ， 而 DelayedTaskConsumer 将 最 “紧急 ” 
的 任务 (到 期 时 间 最 长 的 任务 ) 从 队列 中 取出 ， 然 后 运行 它 。 注 意 ， 这 样 DelayQueue 就 成 为 了 
优先 级 队列 的 一 种 变 体 : 


//: concurrency/DelayQueueDemo. java 
import java.util.concurrent.*; 

import java.util.*; 

import static java.util.concurrent.TimeUnit.*; 
import static net.mindview.utit.Print.*; 


class DelayedTask implements Runnable, Delayed { 
private static int counter = 0; 
private final int id = counter++; 
private final int delta; 
private final long trigger: 
protected static List<DelayedTask> sequence = 
new ArrayList<DelayedTask>(); 
public DelayedTask(int delayInMilliseconds) { 
delta = delayInMiltiseconds; 
trigger = System.nanoTime() + 
NANOSECONDS. convert (delta, MILLISECONDS) ; 
sequence. add( this); 


} 
public long getDelay(TimeUnit unit) { 
return unit.convert( 
trigger - System.nanoTime(), NANOSECONDS); 


} 

public int compareTo(Delayed arg) { 
DelayedTask that = (DelayedTask)arg: 
if(trigger < that.trigger) return -1; 
if(trigger > that.trigger) return 1; 
return 8; 

} 

Public void run() { printnb(this +" "); } 

public String toString() { 
return String. format("{%1$-4d]", delta) + 

"Task " + id; 


} 
public String summary() { 
return "(" + id + ":" + delta + ")": 


public static Class EndSentinel extends DelayedTask { 
private ExecutorService exec: 
public EndSentinel(int delay, ExecutorService e) { 
super (delay) ; 
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exec = 





} 

public void run() { e 
for (DelayedTask pt : sequence) { 

printnb(pt.summary() +" "); 

$ 
printo: 
print(this + " Calling shutdownNow()"); 
exec, shutdownNow() ; 

} 

} 
} 


class DelayedTaskConsumer implements Runnable { 
private DelayQueue<DelayedTask> q; 
public DelayedTaskConsumer (DelayQueve<DelayedTask> q) { 
this.q = q; 


} 
public void run() { 
try { 
while(!Thread. interrupted()) 
q.take().run(); // Run task with the current thread 
} catch(InterruptedException e) { 
// Acceptable way to exit 


} 
print("Finished DelayedTaskConsumer"); 


》 


public class DelayQueueDemo { 
public static void main(String(] args) { 
Random rand = new Random(47); 
ExecutorService exec = Executors.nenCachedThreadPool () ; 
DelayQueue<DelayedTask> queue = 
new DelayQueue<DelayedTask>() ; 
// Fill with tasks that have random delays: 
for(int i = 0; i < 20; i++) 
queue. put (new DelayedTask (rand. next Int (5008) )): 
// Set the stopping point 
queue. add (new DelayedTask.EndSentinel (5000, exec)); 
exec.execute(new DelayedTaskConsumer (queue) ) ; 


} 
) /* Output: 
(128 ] Task 11 [200 } Task 7 {429 ] Task 5 [520] Task 18 
[555 ] Task 1 [961 ] Task 4 [998 ] Task 16 [1267] Task 9 
[1693] Task 2 [1809] Task 14 [1861] Task 3 [2278] Task 15 
[3288] Task 10 [3551] Task 12 [4258] Task @ [4258] Task 19 
[4522] Task 8 [4589] Task 13 [4861] Task 17 [4868] Task 6 
(@:4258) (1:555) (2:1693) (3:1861) (4:961) (5:429) (6:4868) 
(7:200) (8:4522) (9:1207) (10:3288) (11:128) (12:3551) 
(13:4589) (14:1809) (15:2278) (16:998) (17:4861) (18:520) 
(19:4258) (20:5000) 
[5000] Task 20 Calling shutdownNow() 
Finished DelayedTaskConsumer 
“H= 


DelayedTask 包 含 一 个 称 为 sequence 的 List<DelayedTask>， 它 保存 了 任务 被 创建 的 顺序 ， 因 
此 我 们 可 以 看 到 排序 是 按照 实际 发 生 的 顺序 执行 的 。 

Delayed 接 口 有 一 个 方法 名 为 getDelay0， 它 可 以 用 来 告知 延迟 到 期 有 多 长 时 间 ， 或 者 延迟 在 
多 长 时 间 之 前 已 经 到 期 。 这 个 方法 将 强制 我 们 去 使 用 TimeUnit 类 ， 因 为 这 就 是 参数 类 型 。 这 会 
产生 一 个 非常 方便 的 类 ， 因 为 你 可 以 很 容易 地 转换 单位 而 无 需 作 任何 声明 。 例 如 ，delta 的 值 是 
以 毫秒 为 单位 存储 的 ， 但 是 Java SE5 的 方法 System.nanoTime0 产 生 的 时 间 则 是 以 纳 秒 为 单位 的 。 
你 可 以 转换 delta 的 值 ， 方 法 是 声明 它 的 单位 以 及 你 希望 以 什么 单位 来 表示 ， 就 像 下 面 这 样 ; 
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NANOSECONDS. convert (delta, MILLISECONDS); 


在 getDelay0 中 ， 和 希望 使 用 的 单位 是 作为 unit 参 数 传递 进来 的 ， 你 使 用 它 将 当前 时 间 与 触发 
时 间 之 间 的 差 转换 为 调用 者 要 求 的 单位 ， 而 无 需 知道 这 些 单位 是 什么 (这 是 策略 设计 模式 的 一 
个 简单 示例 ， 在 这 种 模式 中 ， 算 法 的 一 部 分 是 作为 参数 传递 进来 的 )。 

为 了 排序 ，Delayed 接 口 还 继承 了 Comparable 接 口 ， 因 此 必须 实现 compareTo0， 使 其 可 以 
产生 合理 的 比较 。toString0 和 summary0 提 供 了 输出 格式 化 ， 而 伐 套 的 EndSentinel 类 提供 了 一 
种 关闭 所 有 事物 的 途径 ， 具 体 做 法 是 将 其 放置 为 队列 的 最 后 一 个 元 素 。 

注意 ， 因 为 DelayedTaskConsumer 自 身 是 一 个 任务 ， 所 以 它 有 自己 的 Thread， 它 可 以 使 用 
这 个 线程 来 运行 从 队列 中 获取 的 所 有 任务 。 由 于 任务 是 按照 队列 优先 级 的 顺序 执行 的 ， 因 此 在 
本 例 中 不 需要 启动 任何 单独 的 线程 来 运行 DelayedTask。 

从 输出 中 可 以 看 到 ， 任 务 创建 的 顺序 对 执行 顺序 没有 任何 影响 ， 任 务 是 按照 所 期 望 的 延迟 
顺序 执行 的 。 

21.7.4 PriorityBlockingQueue 

这 是 一 个 很 基础 的 优先 级 队列 ， 它 具有 可 阻塞 的 读 取 操作 。 下 面 是 一 个 示例 ， 其 中 在 优先 
级 队列 中 的 对 象 是 按照 优先 级 顺序 从 队列 中 出 现 的 任务 。PrioritizedTask 被 赋予 了 一 个 优先 级 数 
字 ， 以 此 来 提供 这 种 顺序 ， 


/1: concurrency/PriorityBlockingQueueDemo .java 
import java.util.concurrent.*; 

import java.util.*; 

import static net.mindview.util.Print.*; 


class PrioritizedTask implements 
Runnable, Comparable<PrioritizedTask> { 
private Random rand = new Random(47); 
private static int counter = 0; 
private final int id = counter++; 
private final int priority; 
protected static List<PrioritizedTask> sequence = 
new ArrayList<PrioritizedTask>(); 
public PrioritizedTask(int priority) { 
this.priority = priority; 
sequence. add(this) ; 


} 
public int compareTo(PrioritizedTask arg) { 
return priority < arg.priority ? 1 : 
(priority > arg.priority ? -1 : 0); 


} 
public void run() { 
try { 
TimeUnit MILLISECONDS. sleep(rand.nextInt(25@)); 
} catch(InterruptedException e) { 
71 Acceptable way to exit 


} 
print(this); 


L 
public String toString() { 
return String. format ("[%1$-3d]", priority) + 
"Task " + id; 


$ 
public String summary() { 
return "(" + id + ":" + priority + ")"; 


} 
public static class EndSentinel extends PrioritizedTask { 
private ExecutorService exec; 
public EndSentinel (ExecutorService e) { 
super(-1); // Lowest priority in this program 











public void run() { 
int count = 0; 
for(PrioritizedTask pt : sequence) { 
printnb(pt.summary()); 
if(++count % 5 == 9) 
print(); 
} 
print(); 
print(this + " Calling shutdownNow()"); 
exec. shutdownNow() ; 
} 
$ 
} 


class PrioritizedTaskProducer implements Runnable { 
private Random rand = new Random(47) ; 
private Queue<Runnable> queue; 
private ExecutorService exec; 
public PrioritizedTaskProducer( 
Queue<Runnable> q, ExecutorService e) { 
queue = q; 
exec = e; // Used for EndSentinel 
} 
public void run() { 
// Unbounded queue; never blocks. 
J) FALL it up fast with random priorities: 
for(int i = 6; 1 < 20; i++) { 
queue.add(new PrioritizedTask(rand.nextint(16))); 
Thread. yield(); 
} 
// Trickle in highest-priority jobs: 
try { 
for(int 1 = @; 1 < 10; i++) { 
TimeUnit MILLISECONDS. sleep (250) ; 
queue. add (new PrioritizedTask(19)); 
, 
7/ Add jobs, lowest priority first: 
for(int 1 = 0; 1 < 10; i++) 
queue. add(new PrioritizedTask(i)); 
// A sentinel to stop all the task: 
queue.add(new PrioritizedTask.EndSentinel (exec)); 
} catch(Interruptedexception e) { 
// Acceptable way to exit 














$ 
print("Finished PrioritizedTaskProducer"): 
} 
} 


class PrioritizedTaskConsumer implements Runnable { 
private PriorityBlockingQueue<Runnable> q; 
public PrioritizedTaskConsumer ( 
PriorityBlockingQueue<Runnable> q) { 
this.q = q; 


try { 
while(!Thread. interrupted()) 
7/ Use current thread to run the task: 
| q.take().run(); 
} catch(Interruptedexception e) { 
// Acceptable way to exit 


} 
print("Finished PrioritizedTaskConsumer"); 


| 
Boblte votd rund) t 
| } 
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public class PriorityBlockingQueueDemo { 

public static void main(String[] args) throws Exception { 
Random rand = new Random(47) ; 
ExecutorService exec = Executors.newCachedThreadPool () ; 
PriorityBlockingQueue<Runnable> queue = 

new PriorityBlockingQueue<Runnable>(); 

exec.execute(new PrioritizedTaskProducer (queue, exec)); 
exec.execute(new PrioritizedTaskConsumer (queue) ) ; 


} 
} /* (Execute to see output) *///:~ 


与 前 一 个 示例 相同 ，PrioritizedTask 对 象 的 创建 序列 被 记录 在 sequence List 中 ， 用 于 和 实际 
的 执行 顺序 比较 。run0 方 法 将 休眠 一 小 段 随机 的 时 间 ， 然 后 打印 对 象 信息 ， 而 EndSentinel 提 供 
了 和 前 面相 同 的 功能 ， 要 确保 它 是 队列 中 最 后 一 个 对 象 。 

PrioritizedTaskProducer 和 PrioritizedTaskComsumer 通 过 PriorityBlockinQueue 彼 此 连接 。 
因为 这 种 队列 的 阻塞 特性 提供 了 所 有 必需 的 同步 ， 所 以 你 应 该 注意 到 了 ， 这 里 不 需要 任何 显 式 
的 同步 一 -不 必 考 虚 当 你 从 这 种 队列 中 读 取 时 ， 其 中 是 否 有 元 素 ， 这 个 队列 在 没有 元 素 时 ， 
将 直接 阻塞 读 取 者 。 
21.7.5 使 用 ScheduledExecutor 的 温室 控制 器 

在 第 10 章 中 介绍 过 可 以 应 用 于 假想 温室 的 控制 系统 的 示例 ， 它 可 以 控制 各 种 设施 的 开关 ， 
或 者 是 对 它们 进行 调节 。 这 可 以 被 看 作 是 一 种 并 发 问题 ， 每 个 期 望 的 温室 事件 都 是 一 个 在 预定 
运行 的 任务 。ScheduledThreadPoolExecutor 提 供 了 解决 该 问题 的 服务 。 通 过 使 用 schedule0) 
(运行 一 次 任务 ) 或 者 scheduleAtFixedRate() (每 隔 规则 的 时 间 重 复 执行 任务 ) ， 你 可 以 将 
Runnable 对 象 设置 为 在 将 来 的 某 个 时 刻 执行 。 将 下 面 的 程序 与 在 第 10 章 中 使 用 的 方式 相 比 ， 就 
会 注意 到 ， 当 你 使 用 像 ScheduledThreadPoolExecutor 这 样 的 预定 义工 具 时 ， 要 简单 许多 : 


//: concurrency/Greenhousescheduler . java 

// Rewriting innerclasses/GreenhouseController. java 
// to use a ScheduledThreadPoolExecutor. 

// {Args: 5080) 

import java.util.concurrent.*; 

import java.util.*; 











public class GreenhouseScheduler { 
private volatile boolean light = false; 
private volatile boolean water = false; 
private String thermostat = "Day"; 
public synchronized String getThermostat() { 
return thermostat; 


} 
public synchronized void setThermostat(String value) { 
thermostat = value; 


} 

ScheduledThreadPoolExecutor scheduler = 
new ScheduledThreadPool Executor (18) ; 

public void schedule(Runnable event, long delay) { 
scheduler . schedule (event „delay ,TimeUnit. MILLISECONDS) ; 


} 
public void 
repeat (Runnable event, long initialDelay, long period) { 
scheduler. scheduleAtFixedRate( 
event, initialDelay, period, TimeUnit MILLISECONDS) ; 


} 
class LightOn implements Runnable { 
public void run() { 
// Put hardware control code here to 
// physically turn on the light. 
System.out.printin(*Turning on lights"): 
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light = true; 
} 
} 
class LightOff implements Runnable { 
public void run() { 
// Put hardware control code here to 
// physically turn off the light. 
System.out.printIn(*Turning off lights"); 
light = false; 
} 
} 
Class WaterOn implements Runnable { 
public void run() { 
// Put hardware control code here. 
System.out.printin("Turning greenhouse water on"); 
water = true; 
} 
} 
Class WaterOff implements Runnable { 
public void run() { 
// Put hardware control code here. 
System.out.printin("Turning greenhouse water off"); 
water = false; 
} 1243] 
$ 
class ThermostatNight implements Runnable { 
public void run() { 
// Put hardware control code here. 
System.out.printin("Thermostat to night setting” 
setThermostat ("Night") ; 
} 
} 
class ThermostatDay implements Runnable { 
public void run() { 
// Put hardware control code here. 
System.out.printin("Thermostat to day setting"); 
setThermostat ("Day"); 
t 

















) 
Class Bell implements Runnable { 
public void run() { System.out.printin("Bing! 
} 
Class Terminate implements Runnable { 
public void run() { 
System.out.println("Terminating"); 
scheduler. shutdownNow() ; 
// Must start a separate task to do this job, 
// since the scheduler has been shut down: 
new Thread() { 
public void run() { 
for(DataPoint d : data) 
System. out.printin(d); 
? 


y.startQ; 
} 
} 
// New feature: data collection 
static class DataPoint { 
final Calendar time; 
final float temperature; 
final float humidity; 
public DataPoint(Calendar d, float temp, float hum) { 
i time = d; 
temperature = temp; 
humidity = hum; 





} 
i public String toString() { 
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return time.getTime() + 
String. format ( 
" temperature: %1$.1f humidity: %2$.2f", 
temperature, humidity); 
} 

} 

private Calendar lastTime = Calendar.getInstance(); 

{ // Adjust date to the half hour 
lastTime. set (Calendar .MINUTE, 30); 
lastTime. set (Calendar. SECOND, 8); 

} 

private float lastTemp = 65.6f; 

private int tempDirection = +1; 

private float lastHumidity = 50.0f; 

private int humidityDirection = +1; 

private Random rand = new Random(47); 

List<DataPoint> data = Collections. synchronizedList( 
new ArrayList<DataPoint>()); 

class CollectData implements Runnable { 
public void run() { 

System.out .printin("Collecting data"); 
synchronized(GreenhouseScheduler.this) { 
// Pretend the interval is longer than it is: 
lastTime. set (Calendar. MINUTE, 
lastTime. get (Calendar .MINUTE) + 30); 
// One in 5 chances of reversing the direction: 
if(rand.nextInt(5) == 4) 
tempDirection = -tempDirection; 
// Store previous value: 
lastTemp = lastTemp + 
tempDirection * (1.0f + rand.nextFloat()); 
if(rand.nextint(5) == 4) 
humidityDirection = -humidityDirection; 
lastHumidity = lastHumidity + 
humidityDirection * rand.nextFloat(): 
// Calendar must be cloned, otherwise all 
// DataPoints hold references to the same LastTime. 
// For a basic object like Calendar, clone() is OK. 
data.add(new DataPoint( (Calendar) lastTime.clone(), 
lastTemp, lastHumidity)); 








} 

} 

public static void main(String{] args) { 
GreenhouseScheduler gh = new GreenhouseScheduler(); 
gh.schedule(gh.new Terminate(), 5000); 
// Former "Restart" class not necessary: 
gh.repeat(gh.new Bell(), @, 1000); 
gh.repeat(gh.new ThermostatNight(), @, 2008); 
gh.repeat(gh.new LightOn(), @, 200) 
gh.repeat(gh.new LightOff(), ©, 480 
gh.repeat(gh.new WaterOn(), ©, 608) 
gh.repeat(gh.new WaterOff(), 6, 890); 
gh.repeat(gh.new ThermostatDay(), @, 1490) 
Bh.repeat(gh.new CollectData(), 560, 509); 








} ;» (Execute to see output) *///:~ 

这 个 版 本 重新 组 织 了 代码 ， 并 且 添加 了 新 的 特性 : 收集 温室 内 的 温度 和 湿度 读数 。DataPoint 
可 以 持 有 并 显示 单个 的 数据 段 ， 而 ColleetData 是 被 调度 的 任务 ， 它 在 每 次 运行 时 ， 都 可 以 产生 仿 
真 数据 ， 并 将 其 添加 到 Greenhouse 的 List<DataPoint> 中 .。 

注意 ，volatile 和 synchronized 在 适当 的 场合 都 得 到 了 应 用 ， 以 防止 任务 之 间 的 互相 干涉 。 在 ' 
持 有 DataPoint 的 List 中 的 所 有 方法 都 是 synchronized 的 ， 这 是 因为 在 List 被 创建 时 ， 使 用 了 
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java.util.Collections 实 用 工具 synchronizedList0 。 

练习 33: (7) 修改 GreenhouseSchedulerjava， 使 其 使 用 DelayQueue 来 替代 Scheduled- 
Executor, 
21.7.6 Semaphore 

正常 的 锁 (来 自 concurrentlocks 或 内 建 的 synchronized 锁 ) 在 任何 时 刻 都 只 允许 一 个 任务 访 
问 一 项 资源 ， 而 计数 信号 量 人 允许 n 个 任务 同时 访问 这 个 资源 。 你 还 可 以 将 信号 量 看 作 是 在 向 外 分 
发 使 用 资源 的 “许可 证 ”， 尽 管 实际 上 没有 使 用 任何 许可 证 对 象 。 

作为 一 个 示例 ， 请 考虑 对 销 池 的 概念 ， 它 管理 着 数量 有 限 的 对 象 ， 当 要 使 用 对 象 时 可 以 签 
出 它们 ， 而 在 用 户 使 用 完毕 时 ， 可 以 将 它们 签 回 。 这 种 功能 可 以 被 封装 到 一 个 泛 型 类 中 : 


//: concurrency/Pool.java 
// Using a Semaphore inside a Pool, to restrict 
// the number of tasks that can use a resource. 
import java.util.concurrent.*; 

import java.util.*; 


public class Pool<T> { 
private int size; 
private List<T> items = new ArrayList<T>(); 
private volatile boolean{] checkedOut; 
private Semaphore available; 
public Pool (Class<T> classObject, int size) { 
this.size = size; 
checkedOut = new bootean[size]; 
available = new Semaphore(size, true); 
// Load pool with objects that can be checked out: 
for(int 1 = @; i < size; ++i) 
try { 
// Assumes a default constructor: 
items. add (classObject .newInstance()); 
} catch(Exception e) { 
throw new RuntimeException(e); 
} 
} 
public T checkOut() throws InterruptedException { 
available. acquire() ; 
return getItem(); 





} 
public void checkIn(T x) { 
if (releaseItem(x)) 
available.release(); 
} 
private synchronized T getitem() { 
for(int i = @; i < size; ++i) 
if(tcheckedOut[i]) { 
checkedOut[i] = true; 
return items.get (i); 


return null; // Semaphore prevents reaching here 


} 
private synchronized boolean releaseltem(T item) { 
int index = items. indexOf (item): 
if(index == -1) return false: // Not in the list 
if(checkedOut{index]) { 
checkedOut [índex] = false; 
return true; 


} 
return false; // Wasn't checked out 


} 
} Mi~ 
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在 这 个 简化 的 形式 中 ， 构 造 器 使 用 newInstance0 来 把 对 象 加 载 到 池 中 。 如 果 你 需要 一 个 新 对 
象 ， 那 么 可 以 调用 checkOut0， 并 且 在 使 用 完 之 后 ， 将 其 递交 给 checkIn0。 

boolean 类 型 的 数组 checkedOut 可 以 跟踪 被 签 出 的 对 象 ， 并 且 可 以 通过 getItem() 和 
releaseItem() 方 法 来 管理 。 而 这 些 都 将 由 Semaphore 类 型 的 available 来 加 以 确保 ， 因 此 ， 在 
checkOut0 中 ， 如 果 没 有 任何 信号 量 许可 证 可 用 (这 意味 着 在 池 中 没有 更 多 的 对 象 了 ) available 
将 阻塞 调用 过 程 。 在 checkIn0 中 ， 如 果 被 签 和 的 对 象 有 效 ， 则 会 向 信号 量 返 回 一 个 许可 证 。 

为 了 创建 一 个 示例 ， 我 们 可 以 使 用 Fat， 这 是 一 种 创建 代价 高 昂 的 对 象 类 型 ， 因 为 它 的 构造 
器 运行 起 来 很 耗 时 : 


//: concurrency/Fat. java 
// Objects that are expensive to create. 


public class Fat { 
private volatile double d; // Prevent optimization 
private static int counter = @; 
private final int id = counter++; 
public Fat() { 
// Expensive, interruptible operation: 
for(int i = 1; i < 10000; i++) { 
d += (Math.PI + Math.£) / (double)i; 
i } 
Public void operation() { System.out.printin(this); } 
public String toString() { return “Fat id: “ + id; } 
} Mi~ 


我 们 在 池 中 管理 这 些 对 象 ， 以 限制 这 个 构造 器 所 造成 的 影响 。 我 们 可 以 创建 一 个 任务 ， 它 
将 签 出 Fat 对 象 ， 持 有 一 段 时 间 之 后 再 将 它们 签 入 ， 以 此 来 测试 Pool 这 个 类 : 


//: concurrency/SemaphoreDeno. java 
// Testing the Pool class 
import java.util.concurrent.*; 
import java.util.*; 
1248) import static net.mindview.util.Print.*; 
// A task to check a resource out of a pool: 
class CheckoutTask<T> implements Runnable { 
private static int counter = 0; 
private final int id = counter++; 
private Pool<T> pool: 
public CheckoutTask(Pool<T> pool) { 
this.pool = pool; 





} 
public void run() { 

try { 
T item = pool.checkOut(); 
-print(this + "checked out "+ item); 
TimeUnit . SECONDS. sleep(1); 
print(this +"checking in ”+ item); 
pool. checkIn(item) ; 
catch(InterruptedException e) { 
// Acceptable way to terminate 
} 


} 
public String toString() { 
return "CheckoutTask "+ id + * "; 
} 
} 
public class SemaphoreDemo { 
final static int SIZE = 25; 
. public static void main(String[] args) throws Exception { 
final Pool<Fat> pool = 
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new Pool<Fat>(Fat.class, SIZE); 
ExecutorService exec = Executors.newCachedThreadPool(); 
for(int i = @; i < SIZE; i++) 
exec. execute (new CheckoutTask<Fat>(pool)); 
print("All CheckoutTasks created"); 
List<Fat> list = new ArrayList<Fat>(); 
; d < SIZE; i++) { 
checkout (); 
main() thread checked out "); 


for(int i = 
Fat f = poo! 
printnb(i + 
f.operation 
Uist.add(f); 








Future<?> blocked = exec.submit(new Runnable() { 
public void run() { 
try { 
// Semaphore prevents additional checkout, 
// so call is blocked: 
pool.checkOut (); 
} catch(InterruptedException e) { 
print("checkOut() Interrupted"); 
$ 
} 


Ye 
TimeUnit . SECONDS. steep(2); 
blocked. cancel (true); // Break out of blocked call 
print("Checking in objects in " + list); 
for(Fat f : list) 
pool. checkIn(f) ; 
for(Fat f : List) 
pool.checkIn(f); // Second checkIn ignored 
exec. shutdown () ; 


} 
} /* (Execute to see output) *///:~ 


在 main0 中 ， 创 建 了 一 个 持 有 Fat 对 象 的 Pool， 而 一 组 CheckoutTask 则 开始 操练 这 个 Pool。 
然后 ，main0 线 程 签 出 地 中 的 Fat 对 象 ， 但 是 并 不 签 入 它们。 一 旦 池 中 所 有 的 对 象 都 被 签 出 ， 
Semaphore 将 不 再 允许 执行 任何 签 出 操作 。blocked 的 run() 方 法 因此 会 被 阻塞 ，2 秒 钟 之 后 ， 
cancel( 方 法 被 调用 ， 以 此 来 挣脱 Future 的 束缚 。 注 意 ， 宛 余 的 签 人 将 被 Pool 忽 略 。 

这 个 示例 依赖 于 Pool 的 客户 端 严格 地 并 愿意 签 和 所持 有 的 对 象 ， 当 其 工作 时 ， 这 是 最 简单 的 
解决 方案 。 如 果 你 无 法 总 是 可 以 依赖 于 此 ，kThinking in Patterns) (7Ewww.MindView.netkh) 深 
人 探讨 了 对 已 经 签 出 对 象 池 的 对 象 的 管理 方式 。 

21.7.7 Exchanger 

Exchanger 是 在 两 个 任务 之 间 交 换 对 象 的 棚 栏 。 当 这 些 任务 进入 栅栏 时 ， 它 们 各 自 拥 有 一 个 
对 象 ， 当 它们 离开 时 ， 它 们 都 拥有 之 前 由 对 象 持 有 的 对 象 。Exchanger 的 典型 应 用 场景 是 : 一 个 
任务 在 创建 对 象 ， 这 些 对 象 的 生产 代价 很 高 昂 ， 而 另 一 个 任务 在 消费 这 些 对 象 。 通 过 这 种 方式 ， 
可 以 有 更 多 的 对 象 在 被 创建 的 同时 被 消费 。 

为 了 演练 Exchanger 类 ， 我 们 将 创建 生产 者 和 消费 者 任务 ， 它 们 经 由 泛 型 和 Generator， 可 
以 工作 于 任何 类 型 的 对 象 ， 然 后 我 们 将 它们 应 用 于 Fat 类 。ExchangerProducer 和 Exchanger- 
Consumer 使 用 一 个 List<T> 作 为 要 交换 的 对 象 ， 它 们 都 包含 一 个 用 于 这 个 List<T> 的 Exchanger。 
当 你 调用 Exchanger.exchanger0 方 法 时 ， 它 将 阻塞 直至 对 方 任务 调用 它 自己 的 exchange0 方 法 ， 
那 时 ， 这 两 个 exchange0 方 法 将 全 部 完成 ， 而 List<T> 则 被 互 换 : 





//: concurrency/ExchangerDemo. java 
import java.util.concurrent.*; 
import java.util.*; 
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import net.mindview.util.*; 


class ExchangerProducer<T> implements Runnable { 
private Generator<T> generator; 
private Exchanger<List<T>> exchanger; 
private List<T> holder: 
Exchanger Producer (Exchanger<List<T>> exchg, 
Generator<T> gen, List<T> holder) { 
exchanger = exchg; 
generator = gen; 
this.holder = holder; 
} 
public void run() { 
try { 
while(!Thread. interrupted()) { 
for(int i = 0; i < ExchangerDemo.size; i++) 
holder .add (generator next ()); 
// Exchange full for empty: 
holder = exchanger . exchange (holder) ; 
} 
} catch(InterruptedException e) { 
// OK to terminate this way. 








} 
} 


class ExchangerConsumer<T> implements Runnable { 
private Exchanger<List<T>> exchanger; 
private List<T> holder; 
private volatile T value; 
ExchangerConsumer (Exchanger<List<T>> ex, List<T> holder) { 
1251 exchanger = ex: 
this.holder = holder; 
} 
public void run() { 
try { 
while(! Thread. interrupted()) { 
holder = exchanger .exchange(holder); 
for(T x : holder) { 
value = x; // Fetch out value 
holder. remove(x); // OK for CopyOnWriteArrayList 
} 
} 
} catch(InterruptedException e) { 
/1 OK to terminate this way. 
} 


System.out.printin("Final value: " + value); 
} 
} 


public class ExchangerDemo { 
static int size = 10; E 
static int delay = 5; // Seconds 
public static void main(String[} args) throws Exception { 
if (args. length > @) 
size = new Integer (args[0]); 
if(args.length > 1) 
delay = new Integer (args{1]); 
ExecutorService exec = Executors.newCachedThreadPool(): 
Exchanger<List<Fat>> xc = new Exchanger<List<Fat>>(): 
List<Fat> 
producerList = new CopyOnwWriteArrayList<Fat>(), 
consumerList = new CopyOnWriteArrayList<Fat>() 7 
exec. execute(new ExchangerProducer<Fat>(xc, 
BasicGenerator.create(Fat.class), producerList)): 
exec.execute( 
new ExchangerConsumer<Fat>(xc,consumerList)); 
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TimeUnit. SECONDS. sleep (delay) ; 
exec. shutdownNow() ; 


} 
} /* Output: (Sample) 
Final value: Fat id: 29999 
Mm 


在 main0 中 ， 创 建 了 用 于 两 个 任务 的 单一 的 Exchanger， 以 及 两 个 用 于 互 换 的 CopyOnWrite- 
ArrayList。 这 个 特定 的 List 变 体 允 许 在 列表 被 遍历 时 调用 remove0 方 法 ， 而 不 会 抛 出 Concurrent- 
ModificationException 异 常 。ExchangeProducer 将 填充 这 个 List， 然 后 将 这 个 满 列表 交换 为 
ExchangerConsumer 传 递 给 它 的 空 列表 。 因 为 有 了 Exchanger， 填 充 一 个 列表 和 消费 另 一 个 列表 
便 可 以 同时 发 生 了 。 

练习 34: (1) 修改 ExchangerDemojava， 让 其 使 用 你 自己 的 类 而 不 是 Fat。 


21.8 仿真 


并 发 最 有 趣 也 最 令 人 兴奋 的 用 法 就 是 创建 仿真 。 通 过 使 用 并 发 ， 仿 真 的 每 个 构件 都 可 以 成 
为 其 自身 的 任务 ， 这 使 得 仿真 更 容易 编程 。 许 多 视频 游戏 和 电影 中 的 CGI 动画 都 是 仿真 ， 前 面 所 
示 的 HorseRacejava 和 GreenhouseSchedulerjava 也 可 以 被 认为 是 仿真 。 

21.8.1 银行 出 纳 员 仿真 

这 个 经 典 的 仿真 可 以 表示 任何 属于 下 面 这 种 类 型 的 情况 ， 对 象 随机 地 出 现 ， 并 且 要 求 由 数 
量 有 限 的 服务 器 提供 随机 数量 的 服务 时 间 。 通 过 构建 仿真 可 以 确定 理想 的 服务 器 数量 。 

在 本 例 中 ， 每 个 银行 顾客 要 求 一 定数 量 的 服务 时 间 ， 这 是 出 纳 员 必须 花费 在 顾客 身上 ， 以 
服务 顾客 需求 的 时 间 单 位 的 数量 。 服 务 时 间 的 数量 对 每 个 顾客 来 说 都 是 不 同 的 ， 并 且 是 随机 确 
定 的 。 另 外 ， 你 不 知道 在 每 个 时 间 间 隔 内 有 多 少 顾客 会 到 达 ， 因 此 这 也 是 随机 确定 的 : 


//: concurrency/BankTellerSimulation. java 
// Using queues and multithreading. 

11 {Args: 5} 

import java.util.concurrent.*; 

import java.util.*: 


// Read-only objects don't require synchronization: 
class Customer { 
private final int serviceTime; 
public Customer(int tm) { serviceTime = tm; } 1253} 
public int getServiceTime() { return serviceTime; } 
public String toString() { 
return "[" + serviceTime + "]"; 
} 
} 














// Teach the customer line to display itself: 
class CustomerLine extends ArrayBlockingQueue<Customer> { 
public CustomerLine(int maxLineSize) { 
super (maxLineSize) ; 


} 
public String toString() { 
if(this.size() == 0) 
return "[Empty]"; 
StringBuilder result = new StringBuilder(): 
for(Customer customer : this) 
result. append (customer); 
return result. toString(); 
} 
k 
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// Randomly add customers to a queue: 
class CustomerGenerator implements Runnable { 
private CustomerLine customers; 
private static Random rand = new Random(47); 
public CustomerGenerator (CustomerLine cq) { 
customers = cq; 
} 
public void run() { 
try ( 
while(!Thread.interrupted()) { 
TimeUnit MILLISECONDS. sleep(rand.nextInt (388)) 
customers. put (new Customer (rand.next Int (1608) ) 
} 
} catch(InterruptedException e) { 


System.out.printin("CustomerGenerator interrupted"); 


$ 
System.out.println("CustomerGenerator terminating” 
} 
¥ 


Class Teller implements Runnable, Comparable<Teller> { 
private static int counter = 
private final int id = counter++; 
// Custémers served during this shift: 
private int customersServed = 0; 
private CustomerLine customers; 
private boolean servingCustomerLine = true; 
public Teller (CustomerLine cq) { customers = cq: } 
public void run() { 
try { 
while(!Thread. interrupted()) { 
Customer customer = customers.take(); 
TimeUnit MILLISECONDS. sleep( 
customer. getServiceTime()) ; 
synchronized(this) { 
customersServed++; 
whi le(!servingCustomerLine) 
wait(); 





} 
} 
} catch(InterruptedException e) { 
System.out.printin(this + "interrupted"); 
} 
System.out.println(this + "terminating"); 
} 
public synchronized void doSomethingElse() { 
customersServed = 0; 
servingCustomerLine = false; 
} 
public synchronized void serveCustomerLine() { 


assert !servingCustomerLine:"already serving: " + this; 


servingCustomerLine = true; 
notifyAll); 
} 3 
Public String toString() { return “Teller "+ id +" 
public String shortString() { return "T" + id; } 
// Used by priority queue: 
public synchronized int compareTo(Teller other) { 
return customersServed < other.customersServed ? - 
(customersServed == other.customersServed ? © 
} 
} 


class TellerManager implements Runnable { 
private ExecutorService exec: 
private CustomerLine customers; 


对 


) 1 





ig 


D; 





发 739 





private PriorityQueue<Teller> workingTellers = 
new Priori tyQueue<Teller>(); 
private Queue<Teller> tellersDoingOtherThings = 
new LinkedList<Teller>(); 
private int adjustmentPeriod: 
private static Random rand = new Random(47); 
public TellerManager (ExecutorService e, 
CustomerLine customers, int adjustmentPeriod) { 
exec = e; 
this.customers = customers; 
this.adjustmentPeriod = adjustmentPeriod; 
// Start with a single teller: 
Teller teller = new Teller(customers); 
exec.execute(teller) ; 
workingTellers.add(teller) ; 
} 
public void adjustTellerNumber() { 
// This is actually a control system. By adjusting 
// the numbers, you can reveal stability issues in 
// the control mechanism. 
// If Line is too long, add another teller: 
if(customers.size() / workingTellers.size() > 2) { 
// If tellers are on break or doing 
// another job, bring one back: 
if (tellersDoingOtherThings.size() > 0) { 
Teller teller = tellersDoingOtherThings.remove(); 
teller.serveCustomerLine(); 
workingTellers.offer (teller); 
return; 


// Else create (hire) a new teller 
Teller teller = new Teller(customers) ; 
exec. execute(teller); 
workingTellers.add(teller); 
return; 
} 
// If line is short enough, remove a teller: 
if (workingTellers.size() > 1 && 
customers.size() / workingTellers.size() < 2) 
reassignOneTeller(): 
// If there is no line, we only need one teller: 
if(customers.size() == 6) 
while(workingTellers.size() > 1) 
reassignOneTeller(); 
) 
// Give a teller a different job or a break: 
private void reassignOneTeller() { 
Teller teller = workingTellers.poll(); 
teller.doSomethingElse(); 
tellersDoingOtherThings. offer (teller); 


} 
public void run() { 
try { 
while(!Thread.interrupted()) { 
TimeUnit MILLISECONDS. sleep (adjustmentPeriod) ; 
adjustTellerNumber () ; 
System.out.print(customers +" { "); 
for(Teller teller : workingTellers) 
System.out.print(teller.shortString() +" "); 

System.out.printin("}"); 


} 
} catch(InterruptedException e) { 
System.out.printin(this + "interrupted"); 


} 
System.out.printin(this + "terminating 
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public String toString() { return "TellerManager "; } 


public class BankTellerSimulation { 
static final int MAX_LINE SIZE = 50; 
static final int ADJUSTHENT_PERIOD = 1000; 
public static void main(String{] args) throws Exception { 
ExecutorService exec = Executors.nenCachedThreadPool () ; 
// If Vine is too long. customers will leave: 
CustomerLine customers = 
new Customerl ine(MAX_LINE_SIZE) ; 
exec. execute (new CustomerGenerator (customers)) ; 
// Manager will add and remove tellers as necessary: 
exec.execute(new TellerHanager ( 
exec, customers, ADJUSTHENT_PERIOD)); 
if(args.length > 6) // Optional argument 
TimeUnit SECONDS. sleep(new Integer (args [9])); 
else { 
System.out.printin("Press ‘Enter’ to quit"); 
System. in. read() ; 


} 

exec. shutdownNow() ; 

} 
} /* Output: (Sample) 
{429} [200] (207] { Te T1 } 
(861) [258] [140] [322] { Te T1 } 
[575] [342] [804] [826] [896] [984] { Te T1 T2 ) 
1384) {816} [141] [12] (689) [992] [976] (368) [395] [354] { T6 T1 

73} 
Teller 2 interrupted 
Teller 2 terminating 
Teller 1 interrupted 
Teller 1 terminating 
TellerManager interrupted 
TellerManager terminating 
Teller 3 interrupted 
Teller 3 terminating 
Teller @ interrupted 
Teller © terminating 
CustomerGenerator interrupted 
CustomerGenerator terminating 
When 


Customer 对 象 非常 简单 ， 只 包含 一 个 final int 域 。 因 为 这 些 对 象 从 来 都 不 发 生变 化 ， 因 此 它 
们 是 只 读 对 象 ， 并 且 不 需要 同步 或 使 用 volatile。 在 这 之 上 ， 每 个 Teller 任 务 在 任何 时 刻 都 只 从 输 
入 队列 中 移 除 一 个 Customer， 并 且 在 这 个 Customer 上 工作 直至 完成 ， 因此 Customer 在 任何 时 刻 
都 只 由 一 个 任务 访问 。 

CustomerLine 表 示 顾 客 在 等 待 被 某 个 Teller 服 务 时 所 排 成 的 单一 的 行 。 这 只 是 一 个 Array- 
BlockingQueue， 它 具有 一 个 toString0 方 法 ， 可 以 按照 我 们 希望 的 形式 打印 结果 。 

CustomerGenerator 附 着 在 CustomerLine 上 ， 按 照 随机 的 时 间 间 隔 向 这 个 队列 中 添加 
Customer, 

Teller 从 CustomerLine 中 取 走 Customer， 在 任何 时 刻 他 都 只 能 处 理 一 个 顾客 ， 并 且 跟 踪 在 这 
个 特定 的 班次 中 有 他 服务 的 Customer 的 数量 。 当 没有 足够 多 的 顾客 时 ， 他 会 被 告知 去 执行 
doSomethingElse0， 而 当 出 现 了 许多 顾客 时 ， 他 会 被 告知 去 执行 serveCustomerLine0)。 为 了 选 
择 下 一 个 出 纳 员 ， 让 其 回 到 服务 顾客 的 业务 上 ， compareTo0 方 法 将 查看 出 纳 员 服务 过 的 顾客 数 
量 ， 使 得 PriorityQueue 可 以 自动 地 将 工作 量 最 小 的 出 纳 员 推 向 前 台 。 

TellerManager 是 各 种 活动 的 中 心 ， 它 跟踪 所 有 的 出 纳 员 以 及 等 待 服务 的 顾客 。 这 个 仿真 中 
有 一 件 有 趣 的 事情 ， 即 它 试图 发 现 对 于 给 定 的 顾客 流 ， 最 优 的 出 纳 员 数量 是 多 少 。 你 可 以 在 
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adjustTellerNumber0 中 看 到 这 一 点 ， 这 是 一 个 控制 系统 ， 它 能 够 以 稳定 的 方式 添加 或 移 除 出 纳 
员 。 所 有 的 控制 系统 都 具有 稳定 性 问题 ， 如 果 它 们 对 变化 反映 过 快 ， 那 么 它们 就 是 不 稳定 的 ， 
而 如 果 它 们 反映 过 慢 ， 则 系统 会 迁移 到 它 的 某 种 极端 情况 。 

练习 35:(8) 修改 BankTellerSimulationjava， 使 它 表 示 Web 客 户 端 ， 向 具有 固定 数量 的 服务 
器 发 送 请 求 。 这 么 做 的 目标 是 要 确定 这 个 服务 器 组 可 以 处 理 的 负载 大 小 。 
21.8.2 饭店 仿真 

这 个 仿真 添加 了 更 多 的 仿真 组 件 ， 例 如 Order 和 Plate， 从 而 充实 了 本 章 前 面 描述 的 
Restaurant.java 示 例 ， 并 且 它 重用 了 第 19 章 中 的 menu 类 。 它 还 引入 了 Java SE5 的 
SynchronousQueue， 这 是 一 种 没有 内 部 容量 的 阻塞 队列 ， 因 此 每 个 putO 都 必须 等 待 一 个 take0， 
反之 亦 然 。 这 就 好 像 是 你 在 把 一 个 对 象 交 给 某 人 一 一 没有 任何 桌子 可 以 放置 这 个 对 象 ， 因 此 只 有 
在 这 个 人 伸 出 手 ， 准 备 好 接收 这 个 对 象 时 ， 你 才能 工作 。 在 本 例 中 ，SynchronousQueue 表 示 设 
置 在 用 餐 者 面前 的 某 个 位 置 ， 以 加 强 在 任何 时 刻 只 能 上 一 道 菜 这 个 概念 。 

本 例 中 剩 下 的 类 和 功能 都 遵循 Restaurantjava 的 结构 ， 或 者 是 对 实际 的 饭店 操作 的 相当 直接 
的 映射 : 

44: concurrency/restaurant2/Restaurantwi thQueues. java 

// {Args: 5} 

package concurrency.restaurant2; 

import enumerated.menu.*; 

import java.util.concurrent.*; 


import java.util.*; 
‘import static net.mindview.util.Print.*; 


// This is given to the waiter, who gives it to the chef: 
class Order { // (A data-transfer object) 
private static int counter = 6; 
private final int id = counter++; 
private’ final Customer customer; 
private final WaitPerson waitPerson; 
private final Food food; 
public Order (Customer cust, WaitPerson wp, Food f) { 
customer = cust; 
waitPerson = wp; 
food = f; 


} 
Public Food item() { return food; } 
public Customer getCustomer() { return customer; } 
public WaitPerson getWaitPerson() { return waitPerson; } 
public String toString() { 
return “Order: "+ id + " item: * + food + 
* for: " + customer + 
" served by: " + waitPerson; 
} 
} 


// This is what comes back from the chef: 
class Plate { 
private final Order order; 
private final Food food; 
public Plate(Order ord, Food f) { 
order = ord; 
food = f; 
} 
public Order getOrder() { return order; } 
public Food getFood() { return food: } 
public String toString() { return food. toString(): } 
} 


class Customer implements Runnable { 
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private static int counter = @; 
private final int id = counter++; 
private final WaitPerson waitPerson: 
// Only one course at a time can be received: 
private SynchronousQueue<Plate> placeSetting = 
new SynchronousQueve<Plate>(); 
public Customer (WaitPerson w) { waitPerson 
public void 
deliver(Plate p) throws InterruptedException { 
// Only blocks if customer is still 
(1260) // eating the previous course: 
placeSetting. put(p) ; 
} 
public void run() { 
for(Course course : Course.values()) { 
Food food = course.randomSelection(); 
try { 
waitPerson.placeOrder(this, food); 
// Blocks until course has been delivered: 
Print(this + "eating " + placeSetting.take()): 
} catch(Interruptedexception e) { 
print(this + “waiting for "+ 
course + " interrupted"); 
break; 
} 
} 
print(this + “finished meal, Leaving"); 
} 
public String toString() { 
return "Customer " + id +" *; 
} 
) 


class WaitPerson implements Runnable { 
private static int counter = @; 
private final int id = counter++; 
private final Restaurant restaurant; 
BlockingQueue<Plate> filledOrders = 
new LinkedBlockingQueue<Plate>(); 











public WaitPerson(Restaurant rest) { restaurant = rest; 


public void placeOrder (Customer cust, Food food) { 
try { 
// Shouldn't actually block because this is 
// a LinkedBlockingQueue with no size Limit: 


restaurant.orders.put(new Order (cust, this, food)): 


} catch(InterruptedException e) { 
print(this + " placeOrder interrupted"); 
$ 
+ 
public void run() { 
try { 
while(! Thread. interrupted()) { 
// Blocks until a course is ready 
Plate plate = filledOrders. take(); 
1261 print(this + "received ”+ plate + 
" delivering to " + 
plate.getOrder().getCustomer()); 
Plate.getOrder().getCustomer() deliver (plate); 





》 
} catch(InterruptedException e) { 
print(this + * interruptes 
} 
print(this + " off duty"); 
} 
public String toString() { 
return “WaitPerson " + id +" *; 


} 





} 








并 发 
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} 


class Chef implements Runnable { 
private static int counter = 0; 
private final int id = counter++; 
private final Restaurant restaurant; 
private static Random rand = new Random(47) 
public Chef(Restaurant rest) { restaurant = rest; } 
public void run() { 
try { 
while(! Thread. interrupted()) { 
// Blocks until an order appears: 
Order order = restaurant.orders.take(): 
Food requestedItem = order. item(); 
// Time to prepare order: 
TimeUnit MILLISECONDS. sleep(rand.nextInt (580); 
Plate plate = new Plate(order, requesteditem) ; 
order .getWaitPerson().filledOrders.put (plate) ; 
} 
} catch(InterruptedException e) { 
print(this +" interrupted"); 
} 
print(this + " off duty"); 
} 
public String toString() { return "Chef "+ id +" "; } 
} 





class Restaurant implements Runnable { 
private List<WaitPerson> waitPersons = 
new ArrayList<WaitPerson>() ; 
private List<Chef> chefs = new ArrayList<Chef>(); 
private ExecutorService exec: 
private static Random rand = new Random(47); 
Block ingQueue<Order> 
orders = new LinkedBlockingQueue<Order>() ; 
public Restaurant (ExecutorService e, int nWaitPersons, 
int nChefs) { 
exec = e; 
for(int 1 = @; 1 < nWaitPersons; i++) { 
WaitPerson waitPerson = new WaitPerson(this): 
wai tPersons.add(waitPerson) ; 
exec. execute (wai tPerson) ; 
} 
for(int 1 =; 1 < nChefs; i++) { 
Chef chef = new Chef(this); 
chefs. add(chef) ; 
exec. execute(chef); 
} 





} 
public void run() { 
try { 
while(!Thread.interrupted()) { 
// A new customer arrives; assign a WaitPerson: 
WaitPerson wp = waitPersons.get( 
rand.nextInt(waitPersons.size())); 
Customer c = new Customer (wp): 
exec.execute(c) ; 
TimeUnit MILLISECONDS. steep(10@) ; 
} 
} catch(InterruptedException e) { 
print("Restaurant interrupted"); 
} 
print("Restaurant closing"); 
} 
+ 


public class RestaurantWithQueues { 
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public static void main(String{] args) throws Exception { 

ExecutorService exec = Executors.newCachedThreadPool(); 
Restaurant restaurant = new Restaurant(exec, 5, 2); 
exec. execute (restaurant) ; 
if(args.length > @) // Optional argument 

TimeUnit. SECONDS. sleep(new Integer (args[9])); 
else { 

print("Press ‘Enter’ to quit"); 

System. in.read(); 


} 
exec. shutdownNow() ; 


} 
} /* Output: (Sample) 
WaitPerson © received SPRING ROLLS delivering to Customer 1 
Customer 1 eating SPRING_ROLLS 
WaitPerson 3 received SPRING ROLLS delivering to Customer 9 
Customer @ eating SPRING_ROLLS 
WaitPerson @ received BURRITO delivering to Customer 1 
Customer 1 eating BURRITO 
WaitPerson 3 received SPRING ROLLS delivering to Customer 2 
Customer 2 eating SPRING ROLLS 
WaitPerson 1 received SOUP delivering to Customer 3 
Customer 3 eating SOUP 
WaitPerson 3 received VINDALOO delivering to Customer 6 
Customer @ eating VINDALOO 
WaitPerson 9 received FRUIT delivering to Customer 1 


“Whim 

关于 这 个 示例 ， 需 要 观察 的 一 项 非常 重要 的 事项 ， 就 是 使 用 队列 在 任务 间 通 信 所 带 来 的 管 
理 复杂 度 。 这 个 单项 技术 通过 反 转 控制 极 大 地 简化 了 并 发 编程 的 过 程 :任务 没有 直接 地 互相 干 
涉 ， 而 是 经 由 队列 互相 发 送 对 象 。 接 收 任务 将 处 理 对 象 ， 将 其 当 作 一 个 消息 来 对 待 ， 而 不 是 向 
它 发 送 消息 。 如 果 只 要 可 能 就 遵循 这 项 技术 ， 那 么 你 构建 出 健壮 的 并 发 系统 的 可 能 性 就 会 大 大 
增加 。 

练习 36: (10) 修改 RestaurantWithQueues.java， 使 得 每 个 桌子 都 有 一 个 OrderTicket 对 象 。 
将 order 修 改 为 orderTickes， 并 添加 一 个 Table 类 ， 每 个 桌子 上 可 以 有 多 个 Customer。 
21.8.3 分 发 工作 

下 面 的 仿真 示例 将 本 章 的 许多 概念 都 结合 在 了 一 起 。 考 虑 一 个 假想 的 用 于 汽车 的 机 器 人 组 
装 线 ， 每 辆 Car 都 将 分 多 个 阶段 构建 ， 从 创建 底盘 开始 ， 紧 跟着 是 安装 发 动机 、 车 病 和 轮子 。 


/1/: concurrency/CarBuilder. java 
// A complex example of tasks working together. 
import java.util.concurrent.*; 

import java.util.*; 

import static net.mindview.util.Print.*; 


class Car { 
private final int id; 
private boolean 
engine = false, driveTrain = false, wheels = false: 
public Car(int idn) { id = idn; } 
// Empty Car object: 
public Car() { id = -1; } 
public synchronized int getId() { return id; } 
public: synchronized void addEngine() { engine = true; } 
public synchronized void addDriveTrain() { 
driveTrain = true; 


public synchronized void addwWheels() { wheels = true; } 
public synchronized String toString() { 
return "Car " + id +" [" + " engine: " + engine 
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+ " drivetrain: "+ drivetrain 
+ "wheels: "+ wheels +" ]"; 
} 
} 


class CarQueue extends LinkedBlockingQueue<Car> {} 


class ChassisBuilder implements Runnable { 
private CarQueue carQueue; 
private int counter = 0; 
public ChassisBuilder(CarQueue cq) { carQueue = cq; } 
public void run() { 
try { 
while(! Thread. interrupted()) { 
TimeUnit MILLISECONDS .sleep(500) ; 
// Make chassis: 
Car c = new Car(counter++); 
print("ChassisBuilder created " + c); 
// Insert into queue 
carQueue. put (c) ; 
} 
} catch(InterruptedException e) { 
print("Interrupted: ChassisBuilder"); 








} 
print("ChassisBuilder off"); 
} 
} 
class Assembler implements Runnable { 
private CarQueue chassisQueue, finishingQueue; 
private Car car; 
private CyclicBarrier barrier = new CyclicBarrier(4); 
private RobotPool robotPool; 
public Assembler (CarQueue cq, CarQueue fq, RobotPool rp){ 
chassisQueue = cq; 
finishingQueue = fq; 
robotPool = rp; 
} 
public Car car() { return car; } 
public CyclicBarrier barrier() { return barrier: } 
public void run() { 
try { 
while(! Thread. interrupted()) { 
// Blocks until chassis is available: 
car = chassisQueue.take(); 
// Hire robots to perform work: 
robotPool .hire(EngineRobot.class, this); 
robotPool.hire(DriveTrainRobot.class, this); 
robotPool.hire(WheelRobot.class, this); 
barrier.await(); // Until the robots finish 
// Put car into finishingQueue for further work 
finishingQueue. put (car); 
} 
} catch(InterruptedException e) { 
print("Exiting Assembler via interrupt"); 
} catch(BrokenBarrierException e) { 
// This one we want to know about 
throw new RuntimeException(e); 





} 
print("Assembler off"); 
} 
$ 


class Reporter implements Runnable { 
private CarQueue carQueue; 
public Reporter (CarQueue cq) { carQueue = cq; } 
public void run() { 
try { 
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while(!Thread.interrupted()) { 
print (carQueue.take()); 
1266] } 


} catch(InterruptedException e) { 
print("Exiting Reporter via interrupt"); 


} 
print("Reporter off"); 





} 
} 


abstract class Robot implements Runnable ( 
private RobotPool pool: 
Public Robot(RobotPool p) { pool = p; } 
protected Assembler assembler; 
Public Robot assignAssembler (Assembler assembler) { 
this.assembler = assembler: 
return this; 
} 
private boolean engage = false; 
public synchronized void engage() { 
engage = true: 
notifyAll(); 
} 
// The part of run() that's different for each robot: 
abstract protected void performService(); 
public void run() { 
try { 
PowerDown(): // Wait until needed 
while(!Thread.interrupted()) { 
performService(); 
assembler.barrier().await(); // Synchronize 
// We're done with that job... 
powerDown(); 
$ 
} catch(InterruptedException e) { 
print("Exiting " + this + * via interrupt"); 
} catch(BrokenBarrierException e) { 
// This one we want to know about 
throw new RuntimeException(e) ; 
} 
print(this + " off"); 
} 
Private synchronized void 
powerDown() throws InterruptedException { 
engage = false; 
assembler = null; // Disconnect from the Assembler 
1267] // Put ourselves back in the available pool: 
pool.release(this); 
while(engage == false) // Power down 
wait(; 
} 


public String toString() { return getClass().getName(); } 
} 





class EngineRobot extends Robot { 
public EngineRobot (RobotPool pool) { super(pool); } 
Protected void performService() { 
print(this + * installing engine"); 
assembler.car().addEngine() ; 
} 
} 


class DriveTrainRobot extends Robot { 
public DriveTrainRobot (RobotPool pool) { super(pool); } 
protected void performService() { 
print(this + " installing DriveTrain"); 
assembler .car().addDriveTrain(); 
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4 
} 


class WheelRobot extends Robot { 
public WheelRobot (RobotPool pool) { super(pool 
Protected void performService() { 
print(this + " installing Wheels"); 
assembler .car () .addwheels(); 
$ 
} 


class RobotPool { 
// Quietly prevents identical entries: A 
private Set<Robot> pool = new HashSet<Robot>(); 
public synchronized void add(Robot r) { 
pool.add(r); 
notifyALl(); 
} 
public synchronized void 
hire(Class<? extends Robot> robotType, Assembler d) 
throws InterruptedException { 
for(Robot r : pool) 
if(r.getClass() .equals(robotType)) { 
pool. remove(r) ; 
1. assignAssembler (d); 
r.engage(); // Power it up to do the task 
return; 
} 
wait(); // None available 
hire(robotType, d): // Try again, recursively 





} 
public synchronized void release(Robot r) { add(r); } 


public class CarBuilder { 
public static void main(String{] args) throws Exception { 
CarQueue chassisQueue = new CarQueue(), 
finishingQueue = new CarQueue(); 
ExecutorService exec = Executors.newCachedThreadPool () ; 
RobotPool robotPool = new RobotPool() ; 
exec.execute(new EngineRobot (robotPool)) ; 
exec. execute(new DriveTrainRobot (robotPool)) ; 
exec.execute(new WheelRobot (robotPool)) : 
exec.execute(new Assembler ( 
chassisQueue, finishingQueue, robotPool)) ; 
exec.execute(new Reporter (fini shingQueue)); 
// Start everything running by producing chassis: 
exec. execute(new Chass isBuilder (chassisQueue)) ; 
TimeUnit. SECONDS. sleep(7); 
exec, shutdownNow() ; 


} VE (Execute to see output) *///:~ 

Car 是 经 由 CarQueue 从 一 个 地 方 传送 到 另 一 个 地 方 的 ，CarQueue 是 一 种 LinkedBlocking- 
Queue 类 型 。ChassisBuilder 创 建 了 一 个 未 加 修饰 的 Car， 并 将 它 放 到 了 一 个 CarQueue 中 。 
Assembler 从 一 个 CarQueue 中 取 走 Car， 并 雇请 Robot 对 其 加 工 。CyclicBarrier 使 Assembler 等 待 ， 
直至 所 有 的 Robot 都 完成 ， 并 且 在 那 一 时 刻 它 会 将 Car 放 置 到 即将 离开 它 的 CarQueue 中 ， 然 后 被 
传送 到 下 一 个 操作 。 最 终 的 CarQueue 的 消费 者 是 一 个 Reporter 对 象 ， 它 只 是 打印 Car， 以 显示 所 
有 的 任务 都 已 经 正确 的 完成 了 。 

Robot 是 在 池 中 管理 的 ， 当 需要 完成 工作 时 ， 就 会 从 池 中 雇请 适当 的 Robot。 在 工作 完成 时 ， 
这 个 Robot 会 返回 到 池 中 。 

在 main0 中 创建 了 所 有 必需 的 对 象 ， 并 初始 化 了 各 个 任务 ， 最 后 启动 ChassisBuilder， 从 而 
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启动 整个 过 程 但是， 由 于 LinkedBlockingQueue 的 行为 使 得 最 先 启动 它 也 没有 问题 )。 注 意 ， 
这 个 程序 遵循 了 本 章 描述 的 所 有 有 关 对 象 和 任务 生命 周期 的 设计 原则 ， 因 此 关闭 这 个 过 程 将 是 
安全 的 。 

你 会 注意 到 ，Car 将 其 所 有 方法 都 设置 成 了 synchronized 的 。 正 如 它 所 表现 出 来 的 那样 ， 在 
本 例 中 ， 这 是 多 余 的 ， 因 为 在 工厂 的 内 部 ，Car 是 通过 队列 移动 的 ， 并 且 在 任何 时 刻 ， 只 有 一 个 
任务 能 够 在 某 辆 车 上 工作 。 基 本 上 ， 队 列 可 以 强制 品行 化 地 访问 Car。 但 是 这 正 是 你 可 能 会 落 入 
的 陷阱 ~ 一 你 可 能 会 说 “让 我 们 尝试 着 通过 不 对 Car 类 同步 来 进行 优化 ， 因 为 看 起 来 Car 在 这 里 
并 不 需要 同步 。” 但 是 稍 后 ， 当 这 个 系统 连接 到 另 一 个 需要 Car 被 同步 的 系统 时 ， 它 就 会 崩溃 。 

Brian Goetz 的 注释 : ` 

进行 这 样 的 声明 会 简单 得 多 :“Car 可 能 会 被 多 个 线程 使 用 ， 因 此 我 们 需要 以 明显 的 方式 使 
其 成 为 线程 安全 的 。” 我 把 这 种 方式 描绘 为 : 在 公园 中 ， 你 会 在 陆 峭 的 坡 路 上 发 现 一 些 保护 围栏 ， 
并 且 可 能 会 发 现 标记 声明 :“ 不 要 侍 冤 围栏 .” 当 然 ， 这 条 规则 的 真实 目的 不 是 要 阻止 你 借助 图 
栏 ， 而 是 防止 你 跌落 遇 虚 。 但 是 “不 要 倚靠 围栏 ”与 “不 要 跌落 二 内 ” WE, RH hd 
要 容易 得 多 的 规则 。 

练习 37: (2) 修改 CarBuilderjava， 在 汽车 构建 过 程 中 添加 一 个 阶段 ， 即 添加 排 气 系统 、 车 
身 和 保险 杠 。 与 第 二 个 阶段 相同 ， 假 设 这 些 处 理 可 以 由 机 器 人 同时 执行 。 

练习 38，(3) 使 用 CarBuilderjava 中 的 方式 ， 对 本 章 中 给 出 的 房屋 构建 过 程 建 模 。 


21.9 性 能 调 优 


在 Java SE5 的 java.util.concurrent 类 库 中 存在 着 数量 庞大 的 用 于 性 能 提高 的 类 。 当 你 细 读 
concurrent 类 库 时 就 会 发 现 很 难 辨 认 哪些 类 适用 于 常规 应 用 (例如 BlockingQueue) ， 而 哪些 类 只 
适用 于 提高 性 能 。 在 本 节 中 ， 我 们 将 围绕 着 性 能 调 优 探讨 某 些 话题 和 类 。 


21.9.1 比较 各 类 互 斥 技术 

既然 Java 包 括 老式 的 synchronized 关 键 字 和 Java SE5 中 新 的 Lock 和 Atomic 类 ， 那 么 比较 这 些 
不 同 的 方式 ， 更 多 地 理解 它们 各 自 的 价值 和 适用 范围 ， 就 会 显得 很 有 意义 。 

比较 天 真 的 方式 是 在 针对 每 种 方式 都 执行 一 个 简单 的 测试 ， 就 像 下 面 这 样 ， 


//:_concurrency/SimpleMicroBenchmark. java 
// The dangers of microbenchmarking. 
import java.util.concurrent.locks.*; 


abstract class Incrementable { 
protected long counter = 0; 
public abstract void increment(); 
} 


class SynchronizingTest extends Incrementable { 
Public synchronized void increment() { ++counter: } 
} 


class LockingTest extends Incrementable { 
private Lock lock = new ReentrantLock(); 
public void increment() { 
lock. Lock() ; 
try { 
++counter; 
} finally { 
lock. unlock () ; 
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} 


public class SimpleMicroBenchmark { 
static long test(Incrementable incr) { 
long start = System.nanoTime(); 
for(long 1 = @; i < 10000000L; i++) 
incr.increment(); 
return System.nanoTime() - start; 
} 
public static void main(String(] args) { 
long synchTime = test(new SynchronizingTest()); 
long lockTime = test(new LockingTest()); 
System. out.printf( "synchronized: ‘%1$16d\n", synchTime); 
System.out.printf("Lock: %1$1@d\n", lockTime) ; 
System. out. printf ("Lock/synchronized = %1$.3f", 
(double) LockTime/ (double) synchTime) ; 


} 
} /* Output: (75% match) 
synchronized: 244919117 
Lock: 939098964 
Lock/synchronized = 3.834 
Wn 


从 输出 中 可 以 看 到 ， 对 synchronized 方 法 的 调用 看 起 来 要 比 使 用 ReentrantLock 快 ， 这 是 为 
什么 呢 ? 

本 例 演示 了 所 谓 的 “ 微 基准 测试 ”危险 ， 这 个 术语 通常 指 在 隔离 的 、 脱 离 上 下 文 环境 的 情 
况 下 对 某 个 特性 进行 性 能 测试 。 当 然 ， 你 仍旧 必须 编写 测试 来 验证 诸如 “Lock 比 synchronized 更 
快 ” 这 样 的 断言 ， 但 是 你 需要 在 编写 这 些 测试 的 时 候 意识 到 ， 在 编译 过 程 中 和 在 运行 时 实际 会 
发 生 什么 。 

上 面 的 示例 存在 着 大 量 的 问题 。 首 先 也 是 最 重要 的 是 ， 我 们 只 有 在 这 些 互 斥 存在 竞争 的 情 
况 下 ， 才 能 看 到 真正 的 性 能 差异 ， 因 此 必须 有 多 个 任务 尝试 着 访问 互 斥 代码 区 。 而 在 上 面 的 示 
例 中 ， 每 个 互 斥 都 是 由 单个 的 main0 线 程 在 隔离 的 情况 下 测试 的 。 

其 次 ， 当 编译 器 看 到 synchronized 关 键 字 时 ， 有 可 能 会 执行 特殊 的 优化 ， 其 至 有 可 能 会 注意 
到 这 个 程序 是 单线 程 的。 编译 器 甚至 可 能 会 识别 出 counter 被 递增 的 次 数 是 固定 数量 的 ， 因 此 会 
预先 计算 出 其 结果 。 不 同 的 编译 器 和 运行 时 系统 在 这 方面 会 有 所 差异 ， 很 难 确切 了 解 将 会 
发 生 什么 ， 但 是 我 们 需要 防止 编译 器 去 预测 结果 的 可 能 性 。 

为 了 创建 有 效 的 测试 ， 我 们 必须 使 程序 更 加 复杂 。 首 先 我 们 需要 多 个 任务 ， 但 并 不 只 是 会 
修改 内 部 值 的 任务 ， 还 包括 读 取 这 些 值 的 任务 (否则 优化 器 可 以 识别 出 这 些 值 从 来 都 不 会 被 使 
用 )。 另 外 ,计算 必须 足够 复杂 和 不 可 预测 ， 以 使 得 编译 器 没有 机 会 执行 积极 优化 。 这 可 以 通过 
预 加 载 一 个 大 型 的 随机 int 数 组 ( 预 加 载 可 以 减 小 在 主 循环 上 调用 Random.nextInt0 所 造成 的 影 
响 )， 并 在 计算 总 和 时 使 用 它们 来 实现 : 








//: concurrency/SynchronizationConparisons .java 
// Comparing the performance of explicit Locks 
// and Atomics versus the synchronized keyword. 
import java.util.concurrent.*; 

import java.util.concurrent.atomic.*; 

import java.util.concurrent.locks.*; 

import java.util.*; 

import static net.mindview.util.Print.*; 


abstract class Accumulator { 


© Brian Goetz 在 向 我 解释 这 些 问题 时 提供 了 很 多 帮助 。 请 查看 在 www:128.ibm.com/developerworks/library/j- 
j 匈 12214 上 的 他 的 文章 ， 以 了 解 更 多 的 性 能 度量 方面 的 知识 。 
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public static long cycles = 50090L; 
// Number of Modifiers and Readers during each test: 
private static final int N = 4; 
public static ExecutorService exec = 
Executors .newF ixedThreadPool (N*2) ; 
private static CyclicBarrier barrier = 
new CyclicBarrier(N*2 + 1); 
protected volatile int index = 0; 
protected volatile long value = 6; 
protected long duration = @; 4 
protected String id = "error"; 
protected final static int SIZE = 100000; 
protected static int[] preloaded = new int [SIZE]; 
static { 
// Load the array of random numbers: 
Random rand = new Random(47) ; 
for(int 1 = @; i < SIZE; i++) 
preLoaded[i] = rand.nextInt(); 
} 
public abstract void accumulate(); 
public abstract long read(); 
private class Modifier implements Runnable { 
public void run() { 
for(long 1 = 0; i < cycles; i++) 
accumulate() ; 
try ( 
barrier.await(); 
} catch(Exception e) { 
throw new RuntimeException(e); 
} 
} 
private class Reader implements Runnable { 
private volatile long value; 
public void run() { 
for(long i = 6; i < cycles; i++) 
value = read(); 
try { 
barrier.await(); 
} catch(Exception e) { 
throw new RuntimeException(e); 
} 
} 
} 
public void timedTest() { 
long start = System.nanoTime(); 
for(int 1 = @; i < N; i++) { 
exec.execute(new Modifier()); 
exec.execute(new Reader ()); 
} 
try { 
barrier.await(); 
} catch(Exception e) { 
throw new Runt imeException(e) ; 
} 
duration = System.nanoTime() - start; 
printf("%-13s: %13d\n", id, duration); 
} 
public static void 
report(Accumulator accl, Accumulator acc2) { 
printf("%-225: %.2f\n", accl.id + "/" + acc2.id, 
(double) accl.duration/ (double) acc2. duration): 
} 


i 











class BaseLine extends Accumulator { 
{ id = "Baseline"; } 
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public void accumulate() { 
value += preloaded [index++] : 
if(index >= SIZE) index = 








} 
public long read() { return value; } 
} 


class SynchronizedTest extends Accumulator { 
{ id = "synchronized"; } 
public synchronized void accumulate() { 
value += preLoaded[index++] ; 
if (index >= SIZE) index = 





} 

public synchronized long read() { 
return value; 

} 


} 


class LockTest extends Accumulator { 
{ id = "Lock"; } 
private Lock lock = new ReentrantLock(); 
public void accumulate() { 
lock. Lock(); $ 
try { 
value += preLoaded|index++) ; 
if(index >= SIZE) index = 
} finally { 
lock. unlock () ; 
} 





} 
public long read() { 
lock, lock(); 


try { 
return value; 
} finally { 


lock.unlock(); 
} 
} 
} 


class AtomicTest extends Accumulator { 

{ id = "Atomic"; } 

private AtomicInteger index = new AtomicInteger (0) ; 

private AtomicLong value = new AtomicLong(@) ; 

public void accumulate() { 
// Oops! Relying on more than one Atomic at 
// a time doesn't work. But it still gives us ， 
// a performance indicator: 
int i = index.getAndIncrement(); 
value. getAndAdd (preLoaded[1] ); 
if (++i >= SIZE) 

index. set (6) ; 








ri 
public long read() { return value.get(): } 
} 


public class SynchronizationComparisons { 
static Baseline baseline = new BaseLine(); 
static SynchronizedTest synch = new SynchronizedTest(); 
static LockTest lock = new LockTest(); 
static AtomicTest atomic = new AtomicTest(); 
static void test() { 
print(" 由 中 
printf("%-12s : %13d\n", "Cycles", Accumulator.cycles); 
baseLine. timedTest(): 
synch. timedTest(); 
lock. timedTest(); 
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atomic, timedTest(); 


Accumulator. 
.report(lock, baseLine); 


Accumulator 


Accumulator. 
Accumulator. 
Accumulator. 
Accumulator. 


} 
public static 


report(synch, baseLine); 


report(atomic, baseline); 
report (synch, lock): 
report (synch, atomic); 
report(lock, atomic); 


void main(String{] args) { 


int iterations = 5; // Default 


if(args. length > @) // Optionally change iterations 


iterations = new Integer (args [0]); 


// The first time fills the thread pool: 


print ("Warmup") ; 
baseLine.timedTest(); 


// Now the initial test doesn't include the cost 
// of starting the threads for the first time. 


41 Produce multiple data points: 























for(int 1 = @; 1 < iterations; i++) { 
test(); 
Accumulator.cycles *= 2; 
} 
Accumulator exec. shutdown() ; 
2 

} /* Output: (Sample) 
Warmup 
BaseLine 34237033 
Cycles 50600 
BaseLine 20966632 
synchronized 24326555 
Lock : 53669950 
Atomic : 30552487 
synchronized/BaseLine : 1.16 
Lock/BaseLine 2.56 
Atomic/BaseLine 1.46 
synchronized/Lock 0.45 
synchronized/Atomic 0.79 
Lock/Atomic 1.76 
Cycles 100000 
BaseLine 41512818 
synchronized : 43843003 
Lock 87430386 
Atomic 51892350 
synchronized/BaseLine : 1.06 
Lock/BaseL ine 2.11 
Atomic/BaseLine 1.25 
synchronized/Lock : 0.50 
synchronized/Atomic 0.84 
Lock/Atomic 1. 









200080 
Baseline 80176676 
synchronized : 5455046661 
Lock 177686829 
Atomic 101789194 
synchronized/BaseLine : 68.04 
Lock/BaseLine : 2.22 
Atomic/BaseLine 21.27 
synchronized/Lock : 30.70 
synchronized/Atomic : 53.59 
Lock/Atomic 1.75 


Cycles 
BaseLine 
synchronized : 






4000900 
160383513 
780052493 
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Lock $ 362187652 
Atomic : 202030984 
synchronized/BaseLine : 4.86 
Lock/BaseLine : 
Atomic/BaseLine 
synchronized/Lock 
synchronized/Atomic 
Lock/Atomic 











cycles 800000 










Basel ine : 322064955 
synchronized : 336155014 
Lock : 704615531 
Atomic. : 393231542 
synchronized/BaseLine : 1.04 
Lock/BaseLine : 
Atomic/BaseLine 
synchronized/Lock 
synchronized/Atomic 
Lock/Atomic 

cycles 1600000 
Baseline : 650004120 
synchronized : 52235762925 
Lock : 1419602771 
Atomic ‘ 796950171 
synchronized/BaseLine : 
Lock/BaseLine : 
Atomic/BaseLine 
synchronized/Lock 


synchronized/Atomic 





Cy 

BaseLine 1285664519 
synchronized : 96336767661 
Lock : 2846988654 
Atomic : 1590545726 ’ 
synchronized/BaseLine : 74.93 
Lock/BaseLine 2 
Atomic/BaseLine : 1.24 
synchronized/Lock : 33.84 
synchronized/Atomic : 68.57 
Lock/Atomic : 1.79 
Whim 


这 个 程序 使 用 了 模版 方法 设计 模式 8 ， 将 所 有 共用 代码 都 放置 到 基 类 中 ， 并 将 所 有 不 同 的 代 
码 隔 离 在 导出 类 的 accumulate0 和 read0 的 实现 中 。 在 每 个 导出 类 SynchronizedTest、LockTest 和 
AtomicTest 中 ， 你 可 以 看 到 accumulate0 和 read0 如 何 表 达 了 实现 互 斥 现象 的 不 同方 式 。 

在 这 个 程序 中 ， 各 个 任务 都 是 经 由 FixedThreadPool 执 行 的 ， 在 执行 过 程 中 尝试 着 在 开始 时 
跟踪 所 有 线程 的 创建 ， 并 且 在 测试 过 程 中 防止 产生 任何 额外 的 开销 。 为 了 保险 起 见 ， 初 始 测试 
执行 了 两 次 ， 而 第 一 次 的 结果 被 丢弃 ， 因 为 它 包含 了 初始 线程 的 创建 。 

程序 中 必须 有 一 个 CyclicBarrier， 因 为 我 们 希望 确保 所 有 的 任务 在 声明 每 个 测试 完成 之 前 都 
已 经 完成 。 

每 次 调用 accumulateO 时 ， 它 都 会 移动 到 preLoaded 数 组 的 下 一 个 位 置 (到达 数组 尾部 时 再 
回 到 开始 位 置 )， 并 将 这 个 位 置 的 随机 生成 的 数字 加 到 value 上 。 多 个 Modifier 和 Reader 任 务 提供 
了 在 Accumulator 对 象 上 的 竞争 。 


日 查看 wwwMindView.net 上 的 《Thinking in Patterns), 





EE) 
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注意 ， 在 AtomicTest 中 ， 我 发 现 情况 过 于 复杂 ， 使 用 Atomic 对 象 已 经 不 适合 了 一 一 基本 上 ， 
如 果 涉 及 多 个 Atomic 对 象 ， 你 就 有 可 能 会 被 强制 要 求 放弃 这 种 用 法 ， 转 而 使 用 更 加 常规 的 互 斥 
(JDK 文 档 特 别 声明 : 当 对 一 个 对 象 的 临界 更 新 被 限制 为 只 涉及 单个 变量 时 ， 只 有 使 用 Atomic 对 象 
这 种 方式 才能 工作 )。 但 是 ， 这 个 测试 仍旧 保留 了 下 来 ， 使 你 能 够 感受 到 Atomic 对 象 的 性 能 优势 。 

在 main0 中 ， 测 试 是 重复 运行 的 ， 并 且 你 可 以 要 求 其 重复 次 数 超过 5 次 (默认 次 数 ) 。 对 于 每 
次 重复 ， 测 试 循环 的 数量 都 会 加 倍 ， 因 此 你 可 以 看 到 当 运行 次 数 越 来 越 多 时 ， 这 些 不 同 的 互 斥 
在 行为 方面 存在 着 怎样 的 差异 。 正 如 你 从 输出 中 可 以 看 到 的 那样 ， 测 试 结果 相当 惊人 。 对 于 前 
四 次 迭代 ，synchronized 关 键 字 看 起 来 比 使 用 Lock 或 Atomic 要 更 高 效 。 但 是 ， 突 然 间 越过 门槛 值 
之 后 ，synehronized 关 键 字 似 平 变 得 非常 低 效 ， 而 Lock 和 Atomic 则 显得 大 体 维持 着 与 BaseLine 测 
试 之 间 的 比例 关系 ， 因 此 也 就 变 得 比 synchronized 关 键 字 要 高 效 得 多 。 

记 住 ， 这 个 程序 只 给 出 了 各 种 互 斥 方式 之 间 的 差异 的 趋势 ,而 上 面 的 输出 也 仅仅 表示 这 些 
差异 在 我 的 特定 环境 下 的 特定 机 器 上 的 表现 。 如 你 所 见 ， 如 果 自 己 动手 试验 ， 当 所 使 用 的 线程 
数量 不 同 ， 或 者 程序 运行 的 时 间 更 长 时 ， 在 行为 方面 肯定 会 存在 着 明显 的 变化 。 例 如 ， 某 些 
hotspot 运 行 时 优化 会 在 程序 运行 数 分 钟 之 后 被 调用 ， 但 是 对 于 服务 器 端 程序 ， 这 段 时 间 可 能 会 
长 达 数 小 时 。 

也 就 是 说 ， 很 明显 ， 使 用 Lock 通 常会 比 使 用 synchronized 要 高 效 许多 ， 而 且 synchronized 的 
开销 看 起 来 变化 范围 太 大 ， 而 Lock 相 对 比较 一 致 。 

这 是 否 意味 着 你 永远 都 不 应 该 使 用 synchronized 关 键 字 呢 ? 这 里 有 两 个 因素 需要 考虑 : 首先 ， 
在 SynchronizationComparisonsjava 中 ， 互 斥 方法 的 方法 体 是 非常 之 小 的 。 通 常 ， 这 是 一 个 很 好 
的 习惯 一 一 只 互 斥 那些 你 绝对 必须 互 斥 的 部 分 。 但 是 ， 在 实际 中 ， 被 互 斥 部 分 可 能 会 比 上 面 示例 
中 的 那些 大 许多 ， 因 此 在 这 些 方法 体 中 花费 的 时 间 的 百分比 可 能 会 明显 大 于 进入 和 退出 互 斥 的 
开销 ， 这 样 也 就 潭 没 了 提高 互 斥 速度 带 来 的 所 有 好 处 。 当 然 ， 唯 一 了 解 这 一 点 的 方式 是 一 当 你 
在 对 性 能 调 优 时 ， 应 该 立即 一 尝试 各 种 不 同 的 方法 并 观察 它们 造成 的 影响 。 

其 次 ， 阅 读本 章 中 的 代码 就 会 发 现 ， 很 明显 ，synchronized 关 键 字 所 产生 的 代码 ， 与 Lock 所 
需 的 “加 锁 -try/finally- 解 锁 ” 惯 用 法 所 产生 的 代码 相 比 ， 可 读 性 提高 了 很 多 ， 这 就 是 为 什么 本 章 
主要 使 用 synchronized 关 键 字 的 原因 。 就 像 我 在 本 书 其 他 地 方 提 到 的 ， 代 码 被 阅读 的 次 数 远 多 于 
被 编写 的 次 数 。 在 编程 时 ， 与 其 他 人 交流 相对 于 与 计算 机 交流 而 言 ， 要 重要 得 多 ， 因 此 代码 的 
可 读 性 至 关 重要 。 因 此 ， 以 synchronized 关 键 字 人 手 ， 只 有 在 性 能 调 优 时 才 赫 换 为 Lock 对 象 这 种 
做 法 ， 是 具有 实际 意义 的 。 

最 后 ， 当 你 在 自己 的 并 发 程序 中 可 以 使 用 Atomic 类 时 ， 这 肯定 非常 好 ， 但 是 要 意识 到 ， 正 
如 我 们 在 SynchronizationComparisonsjava 中 所 看 到 的 ，Atomic 对 象 只 有 在 非常 简单 的 情况 下 才 
有 用 ， 这 些 情况 通常 包括 你 只 有 一 个 要 被 修改 的 Atomic 对 象 ， 并 且 这 个 对 象 独立 于 其 他 所 有 的 
对 象 。 更 安全 的 做 法 是 : 以 更 加 传统 的 互 斥 方式 人 手 ， 只 有 在 性 能 方面 的 需求 能 够 明确 指示 时 ， 
再 替换 为 Atomic。 

21.9.2 免 锁 容器 

就 像 在 第 11 章 中 所 强调 的 ， 容 器 是 所 有 编程 中 的 基础 工具 ， 这 其 中 自然 也 包括 并 发 编程 。 出 
于 这 个 原因 ， 像 Vector 和 Hashtable 这 类 早期 容器 具有 许多 synchronized 方 法 ， 当 它们 用 于 非 多 线 
程 的 应 用 程序 中 时 ， 便 会 导致 不 可 接受 的 开销 。 在 Javal.2 中 ， 新 的 容器 类 库 是 不 同步 的 ， 并 且 
Collections 类 提供 了 各 种 static 的 同步 的 装饰 方法 ， 从 而 来 同步 不 同类 型 的 容器 。 尽 管 这 是 一 种 改 
进 ， 因 为 它 使 你 可 以 选择 在 你 的 容器 中 是 否 要 使 用 同步 ， 但 是 这 种 开销 仍旧 是 基于 synchronized 
加 锁 机 制 的 。Java SE5 特 别 添加 了 新 的 容器 ， 通 过 使 用 更 灵巧 的 技术 来 消除 加 锁 ， 从 而 提高 线程 
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安全 的 性 能 。 

这 些 免 锁 容器 背后 的 通用 策略 是 : 对 容器 的 修改 可 以 与 读 取 操作 同时 发 生 ， 只 要 读 取 者 只 
能 看 到 完成 修改 的 结果 即 可 。 修 改 是 在 容器 数据 结构 的 某 个 部 分 的 一 个 单独 的 副本 (有 时 是 整 
个 数据 结构 的 副本 ) 上 执行 的 ， 并 且 这 个 副本 在 修改 过 程 中 是 不 可 视 的 。 只 有 当 修改 完成 时 ， 
被 修改 的 结构 才 会 自动 地 与 主 数据 结构 进行 交换 ， 之 后 读 取 者 就 可 以 看 到 这 个 修改 了 。 

在 CopyOnWriteArrayList 中 ， 写 人 将 导致 创建 整个 底层 数组 的 副本 ， 而 源 数组 将 保留 在 原 
地 ， 使 得 复制 的 数组 在 被 修改 时 ， 读 取 操 作 可 以 安全 地 执行 。 当 修改 完成 时 ， 一 个 原子 性 的 操 
作 将 把 新 的 数组 换 入 ， 使 得 新 的 读 取 操 作 可 以 看 到 这 个 新 的 修改 。CopyOnWriteArrayList 的 好 
处 之 一 是 当 多 个 迭代 器 同时 遍历 和 修改 这 个 列表 时 , 不 会 抛 出 ConcurrentModificationException， 
因此 你 不 必 编 写 特殊 的 代码 去 防范 这 种 异常 ， 就 像 你 以 前 必须 作 的 那样 。 

CopyOnWriteArraySet 将 使 用 CopyOnWriteArrayList 来 实现 其 免 锁 行为 。 

ConcurrentHashMap 和 ConcurrentLinkedQueue 使 用 了 类 似 的 技术 , 允许 并 发 的 读 取 和 写 入 ， 
但 是 容器 中 只 有 部 分 内 容 而 不 是 整个 容器 可 以 被 复制 和 修改 。 然 而 ， 任 何 修改 在 完成 之 前 ， 读 
取 者 仍旧 不 能 看 到 它们 。ConeurrentHashMap 不 会 抛 出 ConcurrentModificationException 异 常 。 

乐观 锁 

只 要 你 主要 是 从 免 锁 容器 中 读 取 ， 那 么 它 就 会 比 其 synchronized 对 应 物 快 许多 ， 因 为 获取 和 
释放 锁 的 开销 被 省 掉 了 。 如 果 需 要 向 免 锁 容器 中 执行 少量 写 人 ， 那 么 情况 仍旧 如 此 ， 但 是 什么 
算 “ 少 量 ”? 这 是 一 个 很 有 意思 的 问题 。 本 节 将 介绍 有 关 在 各 种 不 同 条 件 下 ， 这 些 容器 在 性 能 
方面 差异 的 大 致 概念 。 

我 将 从 一 个 泛 型 框架 着 手 ， 它 专门 用 于 在 任何 类 型 的 容器 上 执行 测试 ， 包 括 各 种 Map 在 内 ， 
其 中 泛 型 参数 C 表 示 容 器 的 类 型 : 


//: concurrency/Tester.java 
// Framework to test performance of concurrency containers. 
import java.util.concurrent.*; 

import net.mindview.util.*; 


public abstract class Tester<C> { 
static int testReps = 10; 
static int testCycles = 1000; 
static int containerSize = 1000; 
abstract C containerInitiatizer(); 
abstract void startReadersAndwriters(); 
C testContainer: 
String testId; 
int nReaders; 
int AWriters: 
volatile long readResult = 8: 
volatile long readTime = 
volatile long writeTime 
CountDownLatch endLatch; 
static ExecutorService exec = 
Executors.newCachedThreadPool () ; 
Integer {] writeData; 
Tester(String testId, int nReaders, int nWriters) { 
this.testId = testId +" " + 
nReaders + "r " + nWriters + "w"; 
this.nReaders = nReaders; 
this.nWriters = nWriters; 
writeData = Generated. array (Integer.class, 
new RandomGenerator .Integer(), containerSize); 
for(int 1 = @; i < testReps: i++) { 
runTest(); 
readTime = 0; 
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writeTime = 0; 
} 


} 
void runTest() { 
endLatch = new CountDownLatch(nReaders + nWriters); 
testContainer = containerInitializer(); 
startReadersAndwriters(): 
try { 
endLatch.await(): 
} catch(InterruptedException ex) { 
System.out.printIn("endLatch interrupted"); 


$ 
System.out.printt("%-27s %14d %14d\n", 
testId, readTime, writeTime); 
if(readTime != © && writeTime != 6) 
System.out.printf("%-27s %14d\n", 
"readTime + writeTime =", readTime + writeTime); 


abstract class TestTask implements Runnable { 

abstract void test(); 

abstract void putResults(); 

long duration; 

public void run() { 
long startTime = System.nanoTime() ; 
test(); 
duration = System.nanoTime() - startTime; 
synchronized(Tester.this) { 

putResults(); 


} 
endLatch.countDown(); 
} 


} 
1283] public static void initHain(String[] args) { 
if(args.length > 0) 
testReps = new Integer (args [0]); 
if(args.length > 1) 
testCycles = new Integer (args[1]); 
if(args.length > 2) 
containerSize = new Integer(args(2]): 
System.out.printf("%-27s %14s %14s 
"Type", "Read time", “Write time’ 











} 
} Mi~ 


abstract 方 法 containerInitializer0 返 回 将 被 测试 的 初始 化 后 的 容器 ， 它 被 存储 在 testContainer 
域 中 。 另 一 个 abstract 方 法 startReadersAndWriters0 启 动 读 取 者 和 写 人 者 任务 ， 它 们 将 读 取 和 修 
改 待 测 容器 。 不 同 的 测试 在 运行 时 将 具有 数量 变化 的 读 取 者 和 写 人 者 ， 这 样 就 可 以 观察 到 锁 竞争 
(针对 synchronized 容 器 而 言 ) 和 写 人 (针对 免 锁 容器 而 言 ) 的 效果 。 

我 们 向 构造 器 提供 了 各 种 有 关 测 试 的 信息 (参数 标识 符 应 该 是 自 解释 的 )， 然 后 它 会 调用 
runTest0 方 法 repetitions 次 。runTest0 将 创建 一 个 CountDownLatch (因此 测试 可 以 知道 所 有 任 


何 何 时 完成 )、 初 始 化 容器 ， 然 后 调用 startReadersAndWriters0， 并 等 待 它们 全 部 完成 。 


每 个 Reader 和 Writer 类 都 基于 TestTask， 它 可 以 度量 其 抽象 方法 test0 的 执行 时 间 ， 然 后 在 一 


个 synchronized 块 中 调用 putResults0 去 存储 度量 结果 。 - 


为 了 使 用 这 个 框架 (其 中 你 可 以 识别 出 模版 方法 设计 模式 )， 我 们 必须 让 想 要 测试 的 特定 类 


型 的 容器 继承 Tester， 并 提供 适合 的 Reader 和 Writer 类 : 


//: concurrency/ListComparisons. java 
// {Args: 1 10 10} (Fast verification check during build) 
// Rough comparison of thread-safe List performance. 
import java.util.concurrent.*; 

i import java.util.*; 
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import net.mindview.util.*: 


abstract class ListTest extends Tester<List<Integer>> { 
ListTest (String testId, int nReaders, int nWriters) { 
super(testId, nReaders, nWriters); 
} 
class Reader extends TestTask { 
long result = 0; 
void test() { 
for(long i = @; 1 < testCycles; i++) 
for(int index = @; index < containerSize; index++) 
result += testContainer.get (index); 
$ 
void putResults() { 
readResult += result; 
readTime += duration; 
} 
} 
class Writer extends TestTask { 
void test() { 
for(long 1 = 8; 1 < testCycles; i++) 
for(int index = 0; index < containerSize; index++) 
testContainer.set(index, writeDatalindex]) ; 
} 
void putResults() { 
writeTime += duration; 
} 
} n 
void startReadersAndwriters() { 
for(int i = @; i < nReaders; i++) 
exec. execute (new Reader ()) ; 
for(int 1 = 0; i < nWriters; i++) 
exec. execute(new Writer()); 
) 
) 


class SynchronizedArrayListTest extends ListTest { 
List<Integer> ContafnerInitiatizer() { 
return Collections. synchronizedtist( 
new ArrayList<Integer>( 
new CountingIntegerList (containerSize))) ; 
} 
SynchronizedArrayListTest(int nReaders, int nWriters) { 
super ("Synched ArrayList", nReaders, nWriters); 
} 
$ ` 


class CopyOnWriteArrayListTest extends ListTest { 
List<Integer> containerInitializer() { 
return new CopyOnbriteArrayList<Integer>( 
new CountingIntegerList (containerSize)) ; 
} 
CopyOnWriteArrayListTest(int nReaders, int nWriters) { 
Super ("CopyOnWriteArraylist", nReaders, nWriters); 
} 
} 


public class ListComparisons { 
public static void main(String[] args) { 

Tester. initMain(args) ; 
new SynchronizedArrayListTest(18, 6): 
new SynchronizedArrayListTest(9, 1); 
new SynchronizedArrayListTest (5, 5); 
new CopyOnWriteArrayListTest(18, 0 
new CopyOnWriteArrayListTest(9, 1) 
new CopyOnWriteArrayListTest(5, 5); 
Tester .exec. shutdown(); 
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} 
} /* Output: (Sample) 


Type Read time Write time 
Synched ArrayList 10r Ow 23215829470 e 
Synched ArrayList 9r 1w 198947618203 24918613399 
readTime + writeTime = 223866231602 

Synched ArrayList 5r Sw 117367305062 132176613508 
readTime + writeTime = 249543918570 
CopyOnWriteArrayList 19r Ow 758386889 e 
CopyOnWriteArrayList 9r 1w 741305671 136145237 
readTime + writeTime = 877450908 
CopyOnWriteArrayList Sr Sw 212763875 67967464300 
readTime + writeTime = 68180227375 

Wha 


在 ListTest 中 ，Reader 和 Writer 类 执行 针对 List<Integer> 的 具体 动作 。 在 ReaderputResults0 
中 ，duration 被 存储 来 起 来 ，result 也 是 一 样 ， 这 样 可 以 防止 这 些 计算 被 优化 掉 。startReaders- 
AndWriters0 被 定义 为 创建 和 执行 具体 的 Readers 和 Writers。 

一 旦 创建 了 ListTest， 它 就 必须 被 进一步 继承 ， 以 覆盖 containerInitializer()， 从 而 可 以 创建 
和 初始 化 具体 的 测试 容器 。 

在 main0 中 ， 你 可 以 看 到 各 种 测试 变 体 ， 它 们 具有 不 同 数量 的 读 取 者 和 写 人 者 。 由 于 存在 对 
TesterinitMain(args) 的 调用 ， 所 以 你 可 以 使 用 命令 行 参数 来 改变 测试 变量 。 

默认 行 是 为 每 个 测试 运行 1 次， 这 有 助 于 稳定 输出 ， 而 输出 是 可 以 变化 的 ， 因 为 存在 着 诸如 
hotspot 优 化 和 垃圾 回收 这 样 的 JVM 活 动 。 你 看 到 的 样本 输出 已 经 被 编辑 为 只 显示 每 个 测试 的 最 
后 一 个 迭代 。 从 输出 中 可 以 看 到 ，synchronized ArrayList 无 论 读 取 者 和 写 入 者 的 数量 是 多 少 ， 都 
具有 大 致 相同 的 性 能 一 一 读 取 者 与 其 他 读 取 者 竞争 锁 的 方式 与 写 人 者 相同 。 但 是 ，CopyOn- 
WriteArrayList 在 没有 写 人 者 时 ， 速 度 会 快 许 多 ， 并 且 在 有 5 个 写 人 者 时 ， 速 度 仍旧 明显 地 快 。 看 
起 来 你 应 该 尽量 使 用 CopyOnWriteArrayList， 对 列表 写 入 的 影响 并 没有 超过 短期 同步 整个 列表 的 
影响 。 当 然 ， 你 必须 在 你 的 具体 应 用 中 尝试 这 两 种 不 同 的 方式 ， 以 了 解 到 底 哪个 更 好 一 些 。 

再 次 注意 ， 这 还 不 是 测试 结果 绝对 不 变 的 良好 的 基准 测试 ， 你 的 结果 几乎 肯定 是 不 同 的 。 
这 里 的 目标 只 是 让 你 对 两 种 不 同类 型 的 容器 的 相对 行为 有 个 概念 上 的 认识 。 

因为 CopyOnWriteArraySet 使 用 了 CopyOnWriteArrayList， 所 以 它 的 行为 与 此 类 似 ， 在 这 
里 就 不 需要 另外 设计 一 个 单独 的 测试 了 。 

比较 各 种 Map 实 现 

我 们 可 以 使 用 相同 的 框架 来 得 到 synchronizedHashMap 和 ConcurrentHashMap 在 性 能 方面 
的 比较 结果 : 


//: concurrency/MapCompar isons . java 
/1 {Args: 1 10 10} (Fast verification check during build) 
// Rough comparison of thread-safe Map performance. 
import java.util.concurrent.*; 

import java.util.*; 

import net.mindview.util.*; 


abstract class MapTest 
extends Tester<Map<Integer,Integer>> { 
MapTest(String testId, int nReaders, int nWriters) { 
super(testId, nReaders, nWriters); 


} 
class Reader extends TestTask { 


long result = 6; 
void test() { 


日 ”对 于 在 Java 动 态 编译 影响 下 的 基准 测试 的 简介 ， 可 以 查看 www-128.ibm.com/developerworks/library/j-jtp12214。 
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for(long i = 0; i < testCycles; i++) 
for(int index = @; index < containerSize; index++) 
result += testContainer: get (index); 
} 
void putResults() { 
readResult += result; 
readTime += duration; 
} 


} 
class Writer extends TestTask { 


void test() { 
for(long i = @; i < testCycles; i++) 
for(int index = @; index < containerSize; index++) 
testContainer.put(index, writeData[index]); 





} 
void putResults() { 
writeTime += duration; 


} 


} 
void startReadersAndWriters() { 
for(int 1 = @; 1 < nReaders; i++) 
exec.execute(new Reader()); 
for(int i = 0; 1 < nWriters; i++) 
exec.execute(new Writer()): 
} 


} 


class SynchronizedHashMapTest extends MapTest { 
Map<Integer ,Integer> containerInitializer() { 
return Collections. synchronizedMap( 
new HashMap<Integer , Integer>( 
MapData.map( 
new CountingGenerator . Integer (), 
new CountingGenerator. Integer () , 
containerSize))) ; 
} 
SynchronizedHashMapTest(int nReaders, int nWriters) { 
super("Synched HashMap", nReaders, nWriters); 
} 
} 


Class ConcurrentHashMapTest extends HapTest { 
Map<Integer, Integer> containerInitializer() { 
return new ConcurrentHashMap<Integer, Integer>( 
MapData.map( 
new CountingGenerator. Integer (), 
new CountingGenerator.Integer(), containerSize)); 
ConcurrentHashMapTest(int nReaders, int nWriters) { 
super("ConcurrentHashMap", nReaders, nWriters); 
} 
} 


public class HapComparisons { 
public static void main(String{] args) { 
Tester. initMain(args); 
new SynchronizedHashMapTest(18, @) 
new SynchronizedHashMapTest(9, 1 
new SynchronizedHashMapTest (5, 5) 
new ConcurrentHashMapTest(16, ©); 
new ConcurrentHashMapTest (9, 1); 
new ConcurrentHashMapTest (5, 5); 
Tester.exec. shutdown() ; 
} 
} /* Output: (Sample) 
Type Read time Write time 
synched HashMap 10r 9w 306052025049 e 

















1289} 
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Synched HashMap 9r 1w 428319156287 47697347568 
readTime + writeTime = 476016563775 

Synched HashMap Sr Sw 243956877768 244012003202 
readTime + writeTime = 487968880962 
ConcurrentHashMap 10r Ow 23352654318 e 
ConcurrentHashMap 9r 1w 18833089400 1541853224 
readTime + writeTime = 20374942624 
ConcurrentHashMap 5r Sw 12037625732 11850489099 
readTime + writeTime = 23888114831 

Win 


向 ConcurrentHashMap 添 加 写 人 者 的 影响 甚至 还 不 如 CopyOnWriteArrayList 明 显 ， 这 是 因 
为 ConcurrentHashMap 使 用 了 一 种 不 同 的 技术 ， 它 可 以 明显 地 最 小 化 写 人 所 造成 的 影响 。 
21.9.3 乐观 加 锁 

尽管 Atomic 对 象 将 执行 像 decrementAndGet0 这 样 的 原子 性 操作 ， 但 是 某 些 Atomic 类 还 允许 
你 执行 所 谓 的 “乐观 加 锁 "。 这 意味 着 当 你 执行 某 项 计算 时 ， 实 际 上 没有 使 用 互 斥 ， 但 是 在 这 项 
计算 完成 ， 并 且 你 准备 更 新 这 个 Atomic 对 象 时 ， 你 需要 使 用 一 个 称 为 compareAndSet0 的 方法 。 
你 将 旧 值 和 新 值 一 起 提交 给 这 个 方法 ， 如 果 旧 值 与 它 在 Atomic 对 象 中 发 现 的 值 不 一 致 ， 那 么 这 
个 操作 就 失败 一 这 意味 着 某 个 其 他 的 任务 已 经 于 此 操作 执行 期 间 修改 了 这 个 对 象 。 记 住 ， 我 们 
在 正常 情况 下 将 使 用 互 斥 (synehronized 或 Lock) 来 防止 多 个 任务 同时 修改 一 个 对 象 ， 但 是 这 里 
我 们 是 “乐观 的 "， 因 为 我 们 保持 数据 为 未 锁定 状态 ， 并 希望 没有 任何 其 他 任务 插入 修改 它 。 所 
有 这 些 又 都 是 以 性 能 的 名 义 执 行 的 一 通过 使 用 Atomic 来 替代 synchronized 或 Lock， 可 以 获得 性 
能 上 的 好 处 。 

如 果 compareAndSet0 操 作 失 败 会 发 生 什 么 ? 这 正 是 棘手 的 地 方 ， 也 是 你 在 应 用 这 项 技术 时 
的 受 限 之 处 ， 即 只 能 针对 能 够 吻合 这 些 需求 的 问题 。 如 果 compareAndSet0 失 败 ， 那 么 就 必须 决 
定做 些 什么 ， 这 是 一 个 非常 重要 的 问题 ， 因 为 如 果 不 能 执行 某 些 恢复 操作 ， 那 么 你 就 不 能 使 用 这 
项 技术 ， 从 而 必须 使 用 传统 的 互 斥 。 你 可 能 会 重 试 这 个 操作 ， 如 果 在 第 二 次 成 功 ， 那 么 万 事 大 
吉 ， 或 者 可 能 会 忽略 这 次 失败 ， 直 接 结 束 一 一 在 某 些 仿真 中 ,如 果 数 据点 丢失 ,在 重要 的 框架 中 ， 
这 就 是 最 终 需要 做 的 事情 〈 当 然 ， 你 必须 很 好 地 理解 你 的 模型 ， 以 了 解 情况 是 否 确实 如 此 ) 。 

考虑 一 个 假想 的 仿真 ， 它 由 长 度 为 30 的 100000 个 基因 构成 ， 这 可 能 是 某 种 类 型 的 遗传 算法 
的 起 源 。 假 设 伴随 着 遗传 算法 的 每 次 “进化 "， 都 会 发 生 某 些 代价 高 昂 的 计算 ， 因 此 你 决定 使 用 
一 台 多 处 理 器 机 器 来 分 布 这 些 任务 以 提高 性 能 。 另 外 ， 你 将 使 用 Atomic 对 象 而 不 是 Lock 对 象 来 
防止 互 斥 开销 (当然 ， 一 开始 ， 你 使 用 synchronized 关 键 字 以 最 简单 的 方式 编写 了 代码 。 一 旦 你 
运行 该 程序 ， 发 现 它 太 慢 了 ， 并 开始 应 用 性 能 调 优 技术 ， 而 此 时 你 也 只 能 写 出 这 样 的 解决 方案 ) 。 
因为 你 的 模型 的 特性 ， 使 得 如 果 在 计算 过 程 中 产生 冲突 ， 那 么 发 现 冲突 的 任务 将 直接 忽略 它 ， 
并 不 会 更 新 它 的 值 。 下 面 是 这 个 示例 的 代码 : 


//: concurrency/FastSimulation. java 
import java.util.concurrent.*; 

import java.util.concurrent.atomic.*; 
import java.util.*; 

import static net.mindview.util.Print.*; 





public class FastSimutation { 
static final int N_ELEMENTS = 100000; 
static final int N_GENES = 30: 
static final int N_EVOLVERS = 50; 
static final AtomicIntegerf][] GRID = 
new AtomicInteger[N_ELEMENTS] [N_GENES] ; 
static Random rand = new Random(47); 
static class Evolver implements Runnable { 
public void run() { 
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while(!Thread.interrupted()) { 
// Randomly select an element to work ‘on: 
int element = rand.nextInt (N_ELEMENTS) ; 
for(int i = 0; i < N_GENES; i++) { 
int previous = element - 
if (previous < 6) previous = N_ELEMENTS - 1; 
int next = element + 1; 
if(next >= N_ELEMENTS) next = 0; 
int oldvalue = GRID[element] [i] .get(); 





// Perform some kind of modeling calculation: 
int newvalue = oldvalue + 
GRID [previous] [i].get() + GRID[next] [i] .get(); 
newvalue /= 3; // Average the three values 
if (1GRID[element) [i] 
-compareAndSet (oldvalue, newvalue)) { 
J1 Policy here to deal with failure. Here, we 
/1/ just report it and ignore it; our model ` 
// will eventually deal with it. f $ 
print("Old value changed from ”+ oldvalue); 
} 
) ə 
í } 
public static void main(String(] args) throws Exception { 
ExecutorService exec = Executors.newCachedThreadPool () ; 
for(int 1 = 0; i < N_ELEMENTS: i++) 
for(int j = 0; j < N_GENES; j++) 
GRID[4] [j] = new AtomicInteger(rand.nextInt (1080)) ; 
for(int i = @; i < N_EVOLVERS; i++) 
exec.execute(new Evolver()); 
TimeUnit. SECONDS. steep(5); 
exec. shutdownNow() ; 


} 
} /* (Execute to see output) *///:~ 


所 有 元 素 都 被 置 于 数组 内 ， 这 被 认为 有 助 于 提高 性 能 (这 个 假设 将 在 一 个 统 习 中 进行 测试)。 
每 个 Evolver 对 象 会 用 它 前 一 个 元 素 和 后 一 个 元 素来 平均 它 的 值 ， 如 果 在 更 新 时 失败 ， 那 么 将 直 
接 打印 这 个 值 并 继续 执行 。 注 意 ， 在 这 个 程序 中 没有 出 现任 何 互 尺 。 

练习 39，(6) FastSimulation.java 是 否 作出 了 合理 的 假设 ? 试 着 将 数组 从 普通 的 inti 修 改 为 
AtomicInteger， 并 使 用 Lock 互 斥 。 比 较 这 两 个 版 本 的 程序 的 差异 。 

21.9.4 ReadWriteLock 

ReadWriteLock 对 向 数据 结构 相对 不 频繁 地 写 入 ， 但 是 有 多 个 任务 要 经 常 读 取 这 个 数据 结构 
的 这 类 情况 进行 了 优化 。ReadWriteLock 使 得 你 可 以 同时 有 多 个 读 取 者 ， 只 要 它们 都 不 试图 写 人 
即 可 。 如 果 写 锁 已 经 被 其 他 任务 持 有 ， 那 么 任何 读 取 者 都 不 能 访问 ， 直 至 这 个 写 锁 被 释放 为 止 。 

ReadWriteLock 是 否 能 够 提高 程序 的 性 能 是 完全 不 可 确定 的 ， 它 取决 于 诸如 数据 被 读 取 的 频 
率 与 被 修改 的 频率 相 比 较 的 结果 ， 读 取 和 写 入 操作 的 时 间 ( 锁 将 更 复杂 ， 因 此 短 操作 并 不 能 带 
来 好 处 )， 有 多 少 线程 竞争 以 及 是 否 在 多 处 理 机 器 上 运行 等 因素 。 最 终 ， 唯 一 可 以 了 解 
ReadWriteLock 是 否 能 够 给 你 的 程序 带 来 好 处 的 方式 就 是 用 试验 来 证 明 。 

下 面 是 只 展示 了 ReadWriteLock 的 最 基本 用 法 的 示例 : 


//: concurrency/ReaderwriterList. java 
import java.util.concurrent.*; 

import java.util.concurrent.locks.*; 
import java.util.*; 

import static net.mindview.util.Print.*; 


public class ReaderWriterList<T> { 
private ArrayList<T> lockedList; 
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// Make the ordering fair: 
private ReentrantReadwWriteLock lock = 
new ReentrantReadWriteLock(true) ; 
public ReaderWriterList(int size, T initialValue) { 
lockedList = new ArrayList<T>( 
Collections.nCopies(size, initialvalue)); 


} 
public T set(int index, T element) { 
Lock wlock = lock.writeLock(); 
wlock. lock () ; 
try { 
return lockedList.set (index, element); 
} finally { 
wlock.unlock(); 
} 
} 
public T get(int index) { 
Lock rlock = lock.readLock() ; 
rlock. Lock() ; 
try { 
// Show that multiple readers 
// may acquire the read lock: 
if (lock. getReadLockCount() > 1) 
print (lock. getReadLockCount()); 
return lockedList.get(index) ; 
} finally { 
Flock. unlock () ; 
) } 
Public static void main(String[] args) throws Exception { 
new ReaderWriterListTest (30, 1); 
} 
) 


class ReaderWriterListTest { 
ExecutorService exec = Executors.newCachedThreadPool() ; 
private final static int SIZE = 100; 
private static Random rand = new Random(47); 
private ReaderWriterList<Integer> list = 
new ReaderWriterList<Integer>(SIZE, 9); 
private class Writer implements Runnable { 
public void run() { 
try { 
for(int i = 0; 1 < 20; i++) { // 2 second test 
list.set(i, rand.nextInt()); 
TimeUnit . MILLISECONDS. sleep (196) ; 
bi 
} catch(InterruptedException e) { 
// Acceptable way to exit 
下 
print("Writer finished, shutting down"); 
exec. shutdownNow() ; 





} 

4% 

Private class Reader implements Runnable { 
public void run() { 


try { 
while(!Thread.interrupted()) { 
| for(int 1 =: i < SIZE; i++) { 
Uist.get(1); 


TimeUnit.MILLISECONOS. sleep(1); 
} 


} 
| } catch(Interruptedexception e) { 
// Acceptable way to exit 


| Y 
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} 
public ReaderWriterListTest(int readers, int writers) { 
for(int i = 6; i < readers; i++) 
exec.execute(new Reader ()): 
for(int i = 0; i < writers; i++) 
exec. execute(new Writer()); 


} Ne (Execute to see output) *///:~ 

Read WriterList 可 以 持 有 固定 数量 的 任何 类 型 的 对 象 。 你 必须 向 构造 器 提供 所 希望 的 列表 尺 
寸 和 组 装 这 个 列表 时 所 用 的 初始 对 象 。set0 方 法 要 获取 一 个 写 锁 ， 以 调用 底层 的 ArrayList.set0， 
而 get0 方 法 要 获取 一 个 读 锁 ， 以 调用 底层 的 ArrayList.get0。 另 外 ，get0 将 检查 是 否 已 经 有 多 个 
读 取 者 获取 了 读 锁 ， 如 果 是 ， 则 将 显示 这 种 读 取 者 的 数量 ， 以 证 明 可 以 有 多 个 读 取 者 获得 读 锁 。 

为 了 测试 ReaderWriterList，ReaderWriterListTest 为 ReaderWriterList<Integer> 创 建 了 读 取 
者 和 写 和 者。 注意 ， 写 入 者 的 数量 远 少 于 读 取 者 。 1294] 

如 果 你 在 JDK 文 档 中 查看 ReentrantReadWriteLock， 就 会 发 现 还 有 大 量 的 其 他 方法 可 用 ， 
涉及 “公平 性 ”和 “政策 性 决策 ”等 问题 。 这 是 一 个 相当 复杂 的 工具 ， 只 有 当 你 在 搜索 可 以 提 
高 性 能 的 方法 时 ， 才 应 该 想到 用 它 。 你 的 程序 的 第 一 个 草案 应 该 使 用 更 直观 的 同步 ， 并 且 只 有 
在 必需 时 再 引入 ReadWriteLock。 

练习 40，(6) 遵循 ReaderWriterListjava 示 例 ， 使 用 HashMap 创 建 一 个 ReaderWriterMap 。 
通过 修改 MapComparisons.java 来 调查 它 的 性 能 。 它 是 如 何 比较 synchronized HashMap 和 
ConcurrentHashMap 的 ? 


21.10 活动 对 象 


当 你 通读 本 章 之 后 ， 可 能 会 发 现 ，Java 中 的 线程 机 制 看 起 来 非常 复杂 并 难以 正确 使 用 。 另 外 ， 
它 好 像 还 有 点 达 不 到 预期 效果 的 味道 一 -尽管 多 个 任务 可 以 并 行 工作 ， 但 是 你 必须 花 很 大 的 气力 
去 实现 防止 这 些 任务 彼此 互相 干涉 的 技术 。 

如 果 你 曾经 编写 过 汇编 语言 ， 那 么 编写 多 线程 程序 就 似曾相识 : 每 个 细节 都 很 重要 ， 你 有 
责任 处 理 所 有 事物 ， 并 且 没有 任何 编译 器 检查 形式 的 安全 防护 措施 。 

是 多 线程 模型 自身 有 问题 吗 ? 毕竟 ， 它 来 自 于 过 程 型 编程 世界 ， 并 且 几 乎 没 做 什么 改变 。 
可 能 还 存在 着 另 一 种 不 同 的 并 发 模型 ， 它 更 加 适合 面向 对 象 编程 。 

有 一 种 可 替换 的 方式 被 称 为 活动 对 象 或 行动 者 。 之 所 以 称 这 些 对 象 是 “活动 的 "， 是 因为 
每 个 对 象 都 维护 着 它 自己 的 工作 器 线程 和 消息 队列 ， 并 且 所 有 对 这 种 对 象 的 请 求 都 将 进入 队列 
排队 ， 任 何 时 刻 都 只 能 运行 其 中 的 一 个 。 因 此 ， 有 了 活动 对 象 ， 我 们 就 可 以 串 行 化 消息 而 不 是 
方法 ， 这 意味 着 不 再 需要 防备 一 个 任务 在 其 循环 的 中 间 被 中 断 这 种 问题 了 。 

当 你 向 一 个 活动 对 象 发 送 消息 时 ， 这 条 消息 会 转变 为 一 个 任务 ， 该 任务 会 被 插入 到 这 个 对 
象 的 队列 中 ， 等 待 在 以 后 的 某 个 时 刻 运行 。Java SE5 的 Future 在 实现 这 种 模式 时 将 派 上 用 场 。 下 
面 是 一 个 简单 的 示例 ， 它 有 两 个 方法 ， 可 以 将 方法 调用 排 进 队 列 : 


/1: concurrency/Active0bjectDeno. java 
// Can only pass constants, immutables, "disconnected 
// objects." or other active objects as arguments 

// to asynch methods. 

import java.util.concurrent.*; 

import java.util.*; 

import static net.mindview.util.Print.*; 
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public class ActiveObjectDemo { 
private ExecutorService ex = 
Executors .newSingleThreadExecutor (): 
private Random rand = new Random(47) ; 
// Insert a random delay to produce the effect 
// of a calculation time: 
private void pause(int factor) { 
try { 
TimeUnit MILLISECONDS. sleep( 
100 + rand.nextInt(factor)) ; 
} catch(InterruptedException e) { 
print("sleep() interrupted"): 
} 
} 
public Future<Integer> 
calculateInt(final int x, final int y) { 
return ex.submit(new Callable<Integer>() { 
public Integer call() { 
print("starting "+x +*+" +y); 
pause (500) ; 
return x + y; 


D: 
} 
public Future<Float> 
calculateFloat (final float x, final float y) { 
return ex.submit(new Callable<Float>() { 
public Float call() { 
print("starting "+x+" +" +y); 
pause (2608) ; 
return x + y; 
} 
Ð: 


} 
public void shutdown() { ex.shutdown(); } 
public static void main(String[] args) { 
ActiveObjectDemo dl = new ActiveObjectDemo(); 
// Prevents ConcurrentModificattonExcept fon: 
List<Future<?>> results = 
new CopyOnWriteArrayList<Future<?>>( 
for(float f = 0.6f; f < 1.6f; f += @.2f) 
results.add(d1.calculateFloat(f, f)); 
for(int 1 = @; 1 < 5; i++) 
results.add(d1.calculateInt(i, 1)); 
print("All asynch calls made"); 
while(results.size() > 0) { 
for(Future<?> f : results) 
if(f.isDone()) { 
try { 
print(f.get()): 
} catch(Exception e) { 
throw new RuntimeException(e); 
} 


results. remove(f); 





} 
} 
d1. shutdown() ; 
} 

} /* Output: (85% match) 
AIL asynch calls made 
starting 0.6 + 0.0 
starting @.2 + 6.2 
0.9 
starting 0.4 + 0.4 
0.4 
starting 0.6 + 0.6 
0.8 
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starting 0.8 + 6.8 

1.2 

starting 9 + 9 

1.6 

starting 1+1 

9 

starting 2 + 2 

2 

Starting 3 + 3 

4 

starting 4+ 4 

6 

8 

ii~ 

由 对 Executors.newSingleThreadExecutor0 的 调用 产生 的 单线 程 执行 器 维护 着 它 自己 的 无 界 
阻塞 队列 ,并且 只 有 一 个 线程 从 该 队列 中 取 走 任务 并 执行 它们 直至 完成 。 我 们 需要 在 
calculateInt0 和 calculateFloat0 中 做 的 就 是 用 submit0 提 交 一 个 新 的 Callable 对 象 ， 以 响应 对 这 些 
方法 的 调用 ， 这 样 就 可 以 把 方法 调用 转变 为 消息 ， 而 submit0 的 方法 体 包 含 在 匿名 内 部 类 的 call0 
方法 中 。 注 意 ， 每 个 活动 对 象 方法 的 返回 值 都 是 一 个 具有 泛 型 参数 的 Future， 而 这 个 泛 型 参数 就 
是 该 方法 中 实际 的 返回 类 型 。 通 过 这 种 方式 ， 方 法 调用 几乎 可 以 立即 返回 ， 调 用 者 可 以 使 用 
Future 来 发 现 何 时 任务 完成 ， 并 收集 实际 的 返回 值 。 这 样 可 以 处 理 最 复杂 的 情况 ， 但 是 如 果 调用 
没有 任何 返回 值 ， 那 么 这 个 过 程 将 被 简化 。 

在 main0 中 ， 创 建 了 一 个 List<Future<?>> 来 捕获 由 发 送 给 活动 对 象 的 caleulateFloat0 和 
calculateInt0 消 息 返 回 的 Future 对 象 。 对 于 每 个 Future， 都 是 使 用 isDone0 来 从 这 个 列表 中 抽取 的 ， 
这 种 方式 使 得 当 Future 完 成 并 且 其 结果 被 处 理 过 之 后 ， 就 会 从 List 中 移 除 。 注 意 ， 使 用 
CopyOnWriteArrayList 可 以 移 除 为 了 防止 ConcurrentModificationException 而 复制 List 的 这 种 需求 。 

为 了 能 够 在 不 经 意 间 就 可 以 防止 线程 之 间 的 耦合 ， 任 何 传递 给 活动 对 象 方法 调用 的 参数 都 
必须 是 只 读 的 其 他 活动 对 象 ， 或 者 是 不 连接 对 象 (我 的 术语 )， 即 没有 连接 任何 其 他 任务 的 对 象 
(这 一 点 很 难 强制 保障 ， 因 为 没有 任何 语言 支持 它 )。 有 了 活动 对 象 : 

1. 每 个 对 象 都 可 以 拥有 自己 的 工作 器 线程 。 

2. 每 个 对 象 都 将 维护 对 它 自己 的 域 的 全 部 控制 权 (这 上 比 普通 的 类 要 更 严 苛 一 些 ， 普 通 的 类 
只 是 拥有 防护 它们 的 域 的 选择 权 )。 

3. 所 有 在 活动 对 象 之 间 的 通信 都 将 以 在 这 些 对 象 之 间 的 消息 形式 发 生 。 

4. 活动 对 象 之 间 的 所 有 消息 都 要 排队 。 

这 些 结果 很 吸引 人 。 由 于 从 一 个 活动 对 象 到 另 一 个 活动 对 象 的 消息 只 能 被 排队 时 的 延迟 所 
阻塞 ， 并 且 因为 这 个 延迟 总 是 非常 短 且 独立 于 任何 其 他 对 象 的 ， 所 以 发 送 消 息 实 际 上 是 不 可 阻 
塞 的 (最 坏 情况 也 只 是 很 短 的 延迟 )。 由 于 一 个 活动 对 象 系统 只 是 经 由 消息 来 通信 ， 所 以 两 个 对 
象 在 竞争 调用 另 一 个 对 象 上 的 方法 时 ， 是 不 会 被 阻塞 的 ， 而 这 意味 着 不 会 发 生死 锁 。 这 是 一 种 
巨大 的 进步 。 因 为 在 活动 对 象 中 的 工作 器 线程 在 任何 时 刻 只 执行 一 个 消息 ， 所 以 不 存在 任何 资 
源 竞 争 ， 而 你 也 不 必 操心 应 该 如 何 同步 方法 。 同 步 仍旧 会 发 生 ， 但 是 它 通过 将 方法 调用 排队 ， 
使 得 任何 时 刻 都 只 能 发 生 一 个 调用 ， 从 而 将 同步 控制 在 消息 级 别 上 发 生 。 

遗憾 的 是 ， 如 果 没有 直接 的 编译 器 支持 ， 上 面 这 种 编码 方式 实在 是 太 过 于 麻烦 了 。 但 是 ， 
这 在 活动 对 象 和 行动 者 领域 ， 或 者 更 有 趣 的 被 称 为 基于 代理 的 编程 领域 ， 确 实 产生 了 进步 。 代 
理 实际 上 就 是 活动 对 象 ， 但 是 代理 系统 还 支持 跨 网 络 和 机 器 的 透明 性 。 如 果 代理 编程 最 终 成 为 
面向 对 象 编程 的 继任 者 ， 我 一 点 也 不 会 觉得 惊讶 ， 因 为 它 把 对 象 和 相对 容易 的 并 发 解决 方案 结 
合 了 起 来 。 
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通过 搜索 Web， 你 会 发 现 更 多 有 关 活 动 对 象 、 行 动 者 或 代理 的 信息 ， 特 别 是 某 些 在 活动 对 象 
幕后 的 概念 ， 它 们 来 自 C.A.R. Hoare 的 通信 顺序 进程 理论 (Theory of Communicating Sequential 
Processes, CSP) 。 

练习 41: (6) 向 ActiveObjectDemo.java 中 添加 一 个 消息 处 理 器 ， 它 没有 任何 返回 值 ， 在 
main0 调 用 这 个 处 理 器 。 

练习 42: (7) 修改 WaxOMaticjava， 使 其 实现 活动 对 象 。 

作业 。: 使 用 注解 和 Javassist 来 创建 一 个 类 注解 @Active， 将 目标 类 转变 为 活动 对 象 。 


21.11 总 结 


本 章 的 目标 是 向 你 提供 使 用 Java 线 程 进行 并 发 程序 设计 的 基础 知识 ， 以 使 你 理解 : 

1. 可 以 运行 多 个 独立 的 任务 。 

2. 必须 考虑 当 这 些 任务 关闭 时 ， 可 能 出 现 的 所 有 问题 。 

3. 任务 可 能 会 在 共享 资源 上 彼此 干涉 。 互 斥 (B) 是 用 来 防止 这 种 冲突 的 基本 工具 。 

4. 如 果 任务 设计 得 不 够 仔细 ， 就 有 可 能 会 死 锁 。 

明白 什么 时 候 应 该 使 用 并 发 、 什 么 时 候 应 该 避免 使 用 并 发 是 非常 关键 的 。 使 用 它 的 原因 主 
要 是 ， 

“要 处 理 很 多 任务 ， 它 们 交织 在 一 起 ， 应 用 并 发 能 够 更 有 效 地 使 用 计算 机 (包括 在 多 个 CPU 

上 透明 地 分 配 任务 的 能 力 )。 

“要 能 够 更 好 地 组 织 代码 。 

* 要 更 便于 用 户 使 用 。 

均衡 资源 的 经 典 案例 是 在 等 待 输入 /输出 时 使 用 CPU， 更 好 的 代码 组 织 可 以 在 仿真 中 看 到 ， 
使 用 户 方便 的 经 典 案例 是 在 长 时 间 的 下 载 过程 中 监视 “停止 ”按钮 是 否 被 按 下 。 

线程 的 一 个 额外 好 处 是 它们 提供 了 轻 量 级 的 执行 上 下 文 切 换 (大 约 100 条 指令 ) ， 而 不 是 重 
量 级 的 进程 上 下 文 切换 (要 上 千 条 指令 )。 因 为 一 个 给 定 进 程 内 的 所 有 线程 共享 相同 的 内 存 空间 ， 
轻 量 级 的 上 下 文 切 换 只 是 改变 了 程序 的 执行 序列 和 局 部 变量 。 进 程 切换 (重量 级 的 上 下 文 切换 ) 
必须 改变 所 有 内 存 空间 。 

多 线程 的 主要 缺陷 有 ， 

1. 等 待 共享 资源 的 时 候 性 能 降低 。 

2. 需要 处 理 线程 的 额外 CPU 花费 。 

3. 精 糕 的 程序 设计 导致 不 必要 的 复杂 度 。 

4. 有 可 能 产生 一 些 病态 行为 ， 如 饿 死 、 竞 争 、 死 锁 和 活 锁 (多 个 运行 各 自任 务 的 线程 使 得 
整体 无 法 完成 )。 

5. 不 同 平台 导致 的 不 一 致 性 。 比 如 ， 我 在 编写 书 中 的 一 些 例子 时 发 现 ， 竞 争 条 件 在 某 些 机 
器 上 很 快 出 现 ， 但 在 别 的 机 器 上 根本 不 出 现 。 如 果 你 在 后 一 种 机 器 上 做 开发 ， 那 么 当 你 发 布 程 
序 的 时 候 就 要 大 吃 一 惊 了 。 

因为 多 个 线程 可 能 共享 一 个 资源 ， 比 如 一 个 对 象 的 内 存 ， 而 且 你 必须 确定 多 个 线程 不 会 同 
时 读 取 和 改变 这 个 资源 ， 这 就 是 线程 产生 的 最 大 难题 。 这 需要 明智 地 使 用 可 用 的 加 锁 机 制 〈 例 
如 synchronized 关 键 字 ) ， 它 们 仅仅 是 个 工具 ， 同 时 它们 会 引入 潜在 的 死 锁 条 件 ， 所 以 要 对 它们 
有 透彻 的 理解 。 


O 作业 ， 我 建议 读者 将 其 作为 课程 大 作业 。 解 答 指南 中 不 包含 此 类 作业 的 解决 方案 。 
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此 外 ， 线 程 应 用 上 也 有 一 些 技巧 。Java 允许 你 建立 足够 多 的 对 象 来 解决 你 的 问题 ， 至 少 理 
论 上 是 如 此 。( 实 际 上 并 非 如 此 ， 比 如 ， 为 工程 上 的 有 限 元 素 分 析 而 创建 几 百 万 个 对 象 在 Java 中 
如 果 不 使 用 椰 元 设计 模式 ， 是 并 不 可 行 。) 然而 ， 你 要 创建 的 线程 数目 看 起 来 还 是 有 个 上 界 ， 因 
为 达到 了 一 定数 量 之 后 ， 线 程 性 能 会 很 差 。 这 个 临界 点 很 难 检测 ， 通 常 依赖 于 操作 系统 和 
JVM, 它 可 以 是 不 足 一 百 个 线程 ， 也 可 能 是 几 千 个 线程 。 不 过 通常 我 们 只 是 创建 少数 线程 来 解 
决 问题 ， 所 以 这 个 限制 并 不 严重 ， 尽 管 对 于 更 一 般 的 设计 来 说 ， 这 可 能 会 是 一 个 约束 ， 它 可 能 
会 强制 要 求 你 添加 一 种 协作 并 发 模式 。 

不 管 在 使 用 某 种 特定 的 语言 或 类 库 时 ， 线 程 机 制 看 起 来 是 多 么 地 简单 ， 你 都 应 该 视 其 为 魔 
法 。 总 有 一 些 你 最 不 想 碰见 的 事物 会 反 噬 你 一 口 。 哲 学 家 用 餐 问 题 之 所 以 有 趣 ， 就 是 因为 它 可 
以 进行 调整 ， 使 得 死 锁 极 少 发 生 ， 这 给 了 你 一 个 印象 : 每 件 事物 都 很 美好 。 

通常 ， 使 用 线程 机 制 需 要 非常 仔细 和 保守 。 如 果 你 的 线程 问题 变 得 大 而 复杂 ， 那 么 就 应 该 
考虑 使 用 像 Erlang 这 样 的 语言 ， 这 是 专门 用 于 线程 机 制 的 几 种 函数 型 语言 之 一 。 你 可 以 将 这 种 语 
言 用 于 程序 中 要 求 使 用 线程 机 制 的 部 分 ， 前 题 是 你 经 常 要 使 用 线程 机 制 ， 或 者 线程 问题 的 复杂 
度 足 以 促使 你 这 么 做 。 

21.11.1 进 阶 读物 

遗憾 的 是 ， 关 于 并 发 有 大 量 的 误导 信息 一 这 强调 了 它 会 多 么 地 令 人 困惑 ， 以 及 你 会 多 么 轻 
易 地 认为 自己 理解 了 这 些 问题 (我 了 解 这 一 点 ， 因 为 我 自己 有 深刻 的 印象 ， 过 去 我 无 数 次 地 认 
为 自己 已 经 理解 了 线程 机 制 ， 但 是 我 并 不 怀疑 在 将 来 我 还 会 产生 更 多 的 顿悟 之 感 ) 。 当 你 获得 了 
一 篇 关于 并 发 的 新 文献 时 ， 总 是 需要 一 些 警 惕 ， 以 努力 了 解 作者 本 人 理解 哪些 和 不 理解 哪些 。 
下 面 这 些 书籍 ， 我 认为 我 可 以 放心 大 胆 地 说 它们 是 可 靠 的 : 

Kava Concurrency in Practice》， 作 者 Brian Goetz, Tim Peierls, Joshua Bloch, Joseph Bowbeer、 
David Holmes 和 Doug Lea (Addison-Wesley，2006)。 基 本 上 ， 这 就 是 Java 线 程 机 制 世界 中 的 名 人 录 。 

(Concurrent Programming in Java, Second Edition》， 作 者 Doug Lea (Addison-Wesley, 2000), 
尽管 这 本 书 远 早 于 Java SE5， 但 是 Doug 的 许多 成 果 都 成 为 了 新 的 java.util.concurrent 类 库 ， 因 此 
本 书 对 于 全 面 理解 并 发 问题 至 关 重 要 。 它 超越 了 Java 并 发 ， 探 讨 了 当前 跨 语言 和 技术 的 并 发 思想 。 
尽管 在 一 些 地 方 它 显得 有 些 晤 钝 ， 但 是 它 值得 多 次 阅读 (最 好 是 每 次 重读 都 间隔 数 月 ， 这 样 有 
助 于 你 消化 其 中 的 信息 )。Doug 是 确实 理解 并 发 的 少数 人 之 一 ， 因 此 这 是 绝对 值得 一 看 的 书籍 。 

(The Java Language Specification, Third Edition》( 第 17 章 )， 作 者 Gosling、Joy、Steele 和 
Bracha (Addison-Wesley，2005)。 这 是 一 个 技术 规范 ， 更 方便 的 获取 方式 是 到 http://java.sun.com/ 
docs/books/jls 处 下 载 其 电子 文档 。 

所 选 习题 的 答案 都 可 以 在 名 为 The Thinking in Java Annotated Solution Guide 的 电子 文档 中 找 
到 ， 读 者 可 以 从 www.MindView.net 处 购买 此 文档 。 
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第 22 章 图 形 化 用 户 界面 


设计 中 要 遵循 的 一 条 基本 原则 是 :“ 让 简单 的 事情 变 得 容易 ， 让 困难 的 事情 变 得 可 行 .9 

Java 1.0 版 本 中 的 图 形 用 户 界面 (graphical user interface, GUI) 库 ， 其 最 初 的 设计 目标 是 帮 
助 程序 员 编写 在 所 有 平台 上 都 能 良好 表现 的 GUI 程序 。 遗 憾 的 是 ， 这 个 目标 没有 达到 。 事 实 是 
Java 1.0 提 供 的 “抽象 窗口 工具 包 ”(Abstract Window Toolkit, AWT) 在 所 有 的 系统 上 表现 得 都 不 
太 好 ， 而 且 限 制 颇 多 ， 你 只 能 使 用 四 种 字体 ， 也 不 能 访问 存在 于 本 地 操作 系统 上 的 任何 成 熟 的 
GUI 组 件 。Java 1.0 的 AWT 编 程 模型 非常 笨拙 ， 并 且 不 是 面向 对 象 的 。 在 我 课 上 的 一 个 学 生 (在 
Java 创 建 期 间 他 曾经 在 Sun 工 作 ) 解释 了 其 原因 : 最 初版 本 的 AWT 是 在 一 个 月 内 构思 、 设 计 和 实 
现 的 。 从 生产 率 上 看 ， 这 确实 很 惊人 ， 不 过 这 也 是 说 明 为 什么 精心 设计 如 此 重要 的 反面 教材 。 

Java 1.1 的 AWT 中 引入 了 事件 模型 后 (这 是 一 种 更 清晰 的 、 面 向 对 象 的 方法 )， 以 及 随 着 
JavaBeans 的 加 入 ( 它 最 初 是 为 了 使 可 视 化 编程 环境 的 创建 变 得 更 容易 而 引入 的 构件 编程 模型 ) ， 
情况 有 所 好 转 。Java 2 (JDK 1.2) 最 终 完成 了 从 旧式 的 Java 1.0 AWT 到 新 标准 的 转换 :“Java 基 础 类 
库 ”(JFC) 几乎 替换 了 所 有 内 容 ， 其 中 有 关 GUI 的 部 分 被 称 为 “Swing”"。Swing 是 一 组 易于 使 用 、 
易于 理解 的 JavaBeans， 它 能 通过 拖 放 操作 (也 可 以 通过 手工 编写 ) 来 创建 合理 的 GUI 程序 。 软 
件 工 业界 里 的 “三 次 修订 ”规则 (产品 在 修订 三 次 之 后 才 会 成 熟 ) 看 起 来 对 编程 语言 也 同样 适用 。 

本 章 介绍 了 流行 的 Java Swing 库 ， 并 且 合理 地 假定 Swing 就 是 Sun 最 终 的 Java GUI 库 S。 如 果 
出 于 某 些 原因 ， 你 需要 使 用 以 前 那个 “老式 ”的 AWT (比如 你 在 为 以 前 的 代码 做 支持 ， 或 者 由 
于 浏览 器 的 限制 )， 那 么 你 可 以 在 本 书 第 一 版 中 〈 可 以 从 www.BruceEckelcom 下 载 ， 本 书 配套 光 
盘 中 也 有 ) 找到 相关 介绍 。 注 意 ，Java 中 仍然 存在 某 些 AWT 构 件 ， 有 时 你 必须 使 用 它们 。 

请 注意 ， 本 章 没有 完整 地 介绍 Swing 提供 的 构件 ， 对 于 提 到 的 类 ， 也 不 会 讨论 其 所 有 方法 。 
这 里 的 讨论 只 是 一 个 简介 。Swing 库 非常 庞大 ， 本 章 的 目的 仅仅 是 为 你 打 一 个 坚实 的 基础 ， 让 读 
者 熟悉 其 中 的 基本 概念 。 如 果 你 需要 比 这 里 介绍 的 更 复杂 的 功能 ， 只 要 深入 研究 ，Swing 几 乎 可 
以 实现 任何 你 想 要 的 功能 。 

在 这 里 ， 我 假定 你 已 经 从 http://java.sun.com 下 载 并 安装 了 HTML 格 式 的 JDK 文 档 ， 可 以 浏览 
那个 文档 中 的 javax.swing 类 ， 可 以 看 到 完整 的 细节 及 Swing 库 中 的 所 有 方法 。 你 还 可 以 在 Web 上 
搜索 ， 但 是 搜索 的 起 点 最 好 是 http:/java.sun.com/docs/books/tutorial/uiswing 处 Sun 自 己 的 教程 。 

在 学 习 Swing 的 时 候 将 会 发 现 : 

1) Swing 与 其 他 语言 或 开发 环境 相 比 ， 是 一 进 已 经 改进 了 很 多 的 编程 模型 (这 里 并 不 是 说 它 
就 是 完美 的 模型 ， 只 是 说 它 向 前 迈进 了 一 大 步 )。 

2)“GUI 构 造 工具 ”( 可 视 化 编程 环境 ) 对 于 完整 的 Java 开 发 环境 而 言 ， 是 必 不 可 少 的 一 方 
面 。JavaBeans 和 Swing 使 得 GUI 构造 工具 能 够 在 你 用 图 形 工具 向 窗 体 上 放置 组 件 的 同时 帮助 你 编 
写 代码 。 这 不 仅 在 编写 GUI 程 序 期 间 加 快 了 开发 速度 ， 而 且 它 使 得 你 可 以 进行 更 多 的 试验 ， 从 


而 具备 能 够 通过 试验 产生 更 多 设计 的 能 力 ， 继 而 得 到 更 好 的 设计 。 


O 另 一 种 说 法 称 为 “最 小 吃惊 原则 ”"”， 也 就 是 说 ,“ 别 让 用 户 感到 惊讶 。” 
© 注意 ，IBM 公 司 为 其 Eclipse 编辑 器 (www.Eclipse .org) 开发 了 一 套 全 新 的 开源 GUI 库 ， 可 以 把 它 作 为 Swing 之 
外 的 选择 。 本 章 稍 后 会 进行 介绍 。 
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3) Swing 库 设计 上 的 简单 性 和 合理 性 ， 使 得 你 即使 使 用 GUI 构造 工具 而 不 是 手工 编写 代码 ， 
得 到 的 代码 仍然 是 可 读 的 ， 这 就 解决 了 以 前 使 用 GUI 构造 工具 的 一 个 大 问题 ， 就 是 很 容易 产生 
不 可 读 的 代码 。 

Swing 包含 了 所 有 你 希望 在 流行 的 用 户 界面 中 看 到 的 组 件 : 从 带 图 片 的 按钮 ， 到 树 形 和 表格 
组 件 。 这 个 库 虽 然 庞大 ， 但 它 的 设计 理念 是 : 使 用 组 件 的 复杂 程度 与 任务 的 难度 相 匹配 ， 如 果 
任务 很 简单 ， 你 不 用 写 很 多 代码 ， 但 对 于 复杂 的 工作 ， 就 要 写 复杂 的 代码 才 行 。 

Swing 中 有 一 个 非常 令 人 称道 的 原则 ， 称 为 “ 正 交 使 用 ”(orthogonality of use) 。 意 思 是 ， 一 
且 你 理解 了 库 中 的 某 个 通用 概念 ， 你 就 可 以 把 这 个 概念 应 用 到 其 他 地 方 。 比 如 标准 的 命名 约定 ， 
我 在 编写 例子 的 时 候 ， 常 常 在 没有 翻阅 任何 资料 的 情况 下 ， 仅 仅 通过 方法 的 名 称 就 能 正确 猜 出 
其 功能 。 从 库 的 设计 上 来 说 ， 这 是 个 相当 好 的 特性 。 再 比如 ， 通 常 可 以 把 一 个 组 件 “ 插 ”到 另 
一 个 组 件 里 面 ， 而 且 能 正常 工作 。 

Swing 自动 支持 键盘 导航 ， 可 以 不 用 鼠标 运行 Swing 程序 ， 而 且 这 也 不 用 额外 编写 代码 。 要 
支持 滚动 也 不 用 费 工夫 ， 只 要 在 把 组 件 加 入 窗 体 之 前 ， 先 把 它 包装 进 一 个 JScrollPane 组 件 即 可 。 
像 工具 提示 这 样 的 功能 ， 通 常 只 需 一 行 代码 即 可 使 用 。 

为 了 可 移植 性 ，Swing 完 全 用 Java 编 写 。 

Swing 还 支持 一 种 非常 先进 的 功能 ， 称 为 “可 插 式 外 观 ”(pluggable look and feel) ， 意 思 是 
用 户 界面 的 外 观 可 以 动态 改变 ， 以 适应 不 同 平台 和 操作 系统 下 用 户 的 习惯 。 你 甚至 可 以 (不 过 
很 难 ) 自己 发 明 一 种 外 观 。 你 可 以 在 Web 上 找到 一 些 外 观 。 


22.1 applet 


当 Java 刚 面世 时 ， 关 于 它 的 许多 负面 议论 都 来 applet， 它 是 一 种 可 以 在 Internet 上 传递 ， 并 在 
Web 浏 览 器 中 运行 的 程序 (出 于 安全 性 ， 只 能 在 所 谓 的 沙 金 内 运行 )。 人 们 预料 applet 会 成 为 
Internet 演 化 的 下 一 个 阶段 ， 并 且 许 多 Java 方 面 的 原创 书籍 都 认为 人 们 对 Java 感 兴趣 的 原因 就 是 希 
望 能 够 编写 applet。 

由 于 各 种 原因 ， 这 种 革命 并 未 发 生 。 产 生 这 个 问题 的 很 大 一 部 份 原因 在 于 大 多 数 机 器 上 并 
没有 运行 applet 所 必需 的 Java 软 件 ， 而 为 了 运行 某 些 偶然 在 Web 碰 见 的 东西 ， 就 去 下 载 和 安装 
10MB 的 包 对 大 多 数 用 户 来 说 都 是 件 不 情愿 的 事情 。 许 多 用 户 甚至 被 这 种 想法 吓 坏 了 。Java 
applet 作 为 客户 端 应 用 传递 系统 ， 从 来 都 没有 实现 大 规模 应 用 ， 尽 管 你 仍旧 会 偶尔 看 到 applet， 
但 是 实际 上 它们 通常 都 被 于 弃 到 计算 科学 的 特 角 如 网 里 了 。 

然而 ， 这 并 不 意味 着 applet 就 不 是 一 种 有 趣 且 具有 重要 价值 的 技术 。 如 果 你 可 以 保证 用 户 安 
装 了 JRE (例如 在 公司 环境 的 内 部 ) ， 那 么 在 这 种 情况 下 ，applet (或 者 JINLP/Java Web Start， 在 
本 章 稍 后 会 介绍 ) 就 有 可 能 成 为 分 发 客户 程序 和 自动 更 新 所 有 机 器 的 最 佳 方式 ， 而 这 种 方式 不 
需要 分 发 和 安装 新 软件 通常 所 需 的 那些 开销 和 投入 。 4 

你 可 以 在 本 书 的 在 线 支 持 网 站 wwwMindView 上 找到 关于 applet 的 介绍 。 


22.2 Swing 基础 


大 多 数 Swing 应 用 都 被 构建 在 基础 的 JFrame 内 部 ，JFrame 在 你 使 用 的 任何 操作 系统 中 都 可 
以 创建 视窗 应 用 。 视 窗 的 标题 可 以 像 下 面 这 样 使 用 JFrame 的 构造 器 来 设置 : 


O 我 最 喜欢 的 例子 就 是 Ken Amdd 的 “Napkin (餐巾 纸 )” 外 观 ， Eee aa 
请 浏览 htp: //napkinlaf sourceforge.net, 











moa 
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//: gui/Hetloswing. java 
import javax.swing.*; 


public class HelloSwing { 
public static void main(String{] args) { 
JFrame frame = new JFrame("Hello Swing"); 
frame. setDefaultCloseOperation(JFrame.EXIT_ON_CLOSE); 
frame.setSize(300, 100); 
frame.setVisible(true) ; 


} 

} Mh 

setDefaultCloseOperation() 告 诉 JFrame 当 用 户 执行 关闭 操作 时 应 该 做 些 什么 。EXIT_ 
ON_CLOSE 常 量 告诉 它 要 退出 程序 。 如 果 没 有 这 个 调用 ， 默 认 的 行为 是 什么 也 不 做 ， 因 此 应 用 
将 不 会 关闭 。setSize0 以 像素 为 单位 设置 视窗 的 尺寸 。 请 注意 最 后 一 行 : 

frame. setVisible(true) ; 
如 果 没 有 这 行 ， 你 在 屏幕 上 将 什么 也 看 不 到 。 

我 们 可 以 通过 在 JFrame 中 添加 一 个 JLabel 来 使 事情 变 得 更 有 趣 一 些 ; 


//: Bui/HelloLabel.java 
import javax.swing.*; 
import java.util.concurrent.*; 


publié class HelloLabel { 
public static void main(String{] args) throws Exception { 
JFrame frame = new JFrame("Hello Swing"); 
JLabel label = new JLabel("A Label"): 
frame.add(1abel) ; 
frame. setDefaul tCloseOperation( JFrame. EXIT_ON_CLOSE) ; 
frame.setSize(300, 100); 
frame. setVisible(true) ; 
TimeUnit SECONDS. sleep(1); 
label.setText("Hey! This is Different!"); 


} 
yh 


在 一 秒 钟 之 后 ，JLabel 的 文本 发 生 了 变化 。 尽 管 这 对 于 这 个 小 程序 来 说 既 有 趣 又 安全 ， 但 
是 对 于 main() 线 程 来 说 ， 直 接 对 GUI 组 件 编写 代码 并 非 是 一 种 好 的 想法 。Swing 有 它 自 己 的 专用 
线程 来 接收 UI 事件 并 更 新 屏幕 ， 如 果 你 从 其 他 线程 着 手 对 屏幕 进行 操作 ， 那 么 就 可 能 会 产生 第 
21 章 中 所 描述 的 冲突 和 死 锁 。 

取而代之 的 是 ， 其 他 线程 ， 例 如 这 里 是 像 main0 这 样 的 线程 ， 应 该 通过 Swing 事件 分 发 线程 
提交 要 执行 的 任务 。 你 可 以 通过 将 任务 提交 给 SwingUtilities.invokeLater0 来 实现 这 种 方式 ， 
这 个 方法 会 通过 事件 分 发 线程 将 任务 放置 到 (最终 将 得 到 执行 的 ) 待 执行 事件 队列 中 。 如 果 我 
们 将 这 种 方式 应 用 于 上 面 的 示例 ， 那 么 它 就 会 变 成 下 面 的 样子 : 

//: gui/SubmitLabelManipulationTask.java 


import javax.swing.*; 
import java.util.concurrent.*; 


public class SubmitLabelManipulationTask { 
public static void main(String{] args) throws Exception { 
JFrame frame = new JFrame("Hello Swing"): 
final JLabel label = new JLabel("A Label"): 
frame. add(label); 
frame. setDefaultCloseOperation(JFrame.EXIT_ON_CLOSE) ; 
frame.setSize(309, 160); 
frame.setVisible(true); 


O 从 技术 上 讲 ,事件 分 发 线程 来 自 AWT 类 库 。 





BYRAP RG 771 





TimeUnit. SECONDS. sleep(1); 
SwingUtilities. invokeLater(new Runnable() { 
Public void run() { 
label.setText("Hey! This is Different!"); 
} 
De 


} 

} M~ 

现在 你 再 也 不 用 直接 操作 JLabel 了 。 取 而 代 之 的 是 ， 你 提交 一 个 Runnable， 当 事件 分 发 线 
程 在 事件 队列 中 获取 这 项 任务 时 ， 它 将 执行 实际 的 操作 ， 并 且 在 执行 这 个 Runnable 时 ， 不 会 做 
其 他 任何 事情 ， 因 此 也 就 不 会 产生 任何 冲突 ， 当 然 ， 前 提 是 程序 中 的 所 有 代码 都 遵循 这 种 通过 
SwingUtilities.invokeLater0 来 提交 操作 的 方式 。 这 包括 启动 程序 自身 ， 即 main0 也 不 应 该 调用 
Swing 的 方法 ， 就 像 上 面 的 程序 一 样 ， 它 应 该 向 事件 队列 提交 任务 8 。 因 此 ， 所 编写 的 恰当 的 程 
序 看 起 来 应 该 是 下 面 的 样子 : 

//: gui/Submi tswingProgram. java 


import javax.swing.*; 
import java.util.concurrent.*; 


public class SubmitSwingProgram extends JFrame { 
JLabel label; 
public SubmitSwingProgram() { 
super("Hello Swing"); 
label = new JLabel("A Label"); 
add(label) ; 
setDefaultCloseOperation (JFrame. £XIT_ON_CLOSE) ; 
setSize(300, 190); 
setVisible(true) ; 
} 
static SubmitSwingProgram ssp; 
public static void main(String{] args) throws Exception { 
SwingUtilities. invokeLater(new Runnable() { 
Public void run() { ssp = new SubmitSwingProgram(); } 
Ys 
TimeUnit. SECONDS. sleep(1); 
SwingUtilities. invokeLater (new Runnable() { 
Public void run() { 
Ssp.label.setText("Hey! This is Different!"); 
} 
H: 


} 
dh 


注意 ， 对 sleep0 的 调用 不 在 构造 器 的 内 部 。 如 果 你 将 它 放 在 构造 器 内 部 ，JLabel 的 初始 文本 
就 永远 都 不 会 出 现 。 这 主要 是 因为 构造 器 在 sleep0 调 用 完毕 和 新 的 标签 插入 之 前 不 会 结束 ， 如 
果 sleep0 在 构造 器 的 内 部 ， 或 者 在 任何 UI 操作 的 内 部 ， 那 么 就 意味 着 你 在 sleep0 期 间 将 中 止 事件 
分 发 线程 ， 这 通常 是 个 精 炎 的 主意 。 

练习 1: (1) 修改 HelloSwingjava， 向 你 自己 证 明 如 果 没 有 对 setDefaultCloseOperation0 的 调 
用 ， 应 用 程序 就 不 会 关闭 。 

练习 2: (2) 修改 HelloSwingjava， 通 过 添加 随机 数量 的 标签 ， 说 明 标签 的 添加 是 动态 的 。 
22.2.1 一 个 显示 框架 

我 们 可 以 创建 一 个 显示 框架 ， 将 其 用 于 本 章 剩余 部 分 的 Swing 示例 中 ， 从 而 使 得 上 面 的 想法 
得 以 结合 ， 并 减少 了 元 余 代码 

日 这 个 实践 被 添加 到 了 Java SE5 中 ， 因 此 你 在 很 多 旧 程 序 中 看 到 它们 并 没有 这 么 做 。 这 并 不 意味 着 这 些 程序 的 编 

写 者 忽视 了 这 一 点 。 建 议 性 的 实践 看 起 来 在 不 断 地 演化 。 
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//: net/mindview/util/SwingConsole. java 
// Tool for running Swing demos from the 
// console, both applets and JFrames. 
package net.mindview.util: 

import javax.swing.*; 


public class SwingConsole { 
public static void 
run(final JFrame f, final int width, final int height) { 
“SwingUtilities.invokeLater(new Runnable() { 
public void run() { 
f.setTitle(f.getClass().getSimpleName()) ; 
f.setDefaul tCloseOperat ion (JFrame. EXIT_ON CLOSE): 
f.setSize(width, height); 
f.setVisible(true); 
} 
H: 


} 
dh 
这 可 能 是 一 个 你 想 要 自己 使 用 的 工具 ， 因 此 它 被 放 到 了 net.mindview.util 类 库 中 。 要 想 使 用 
它 ， 你 的 应 用 就 必须 位 于 一 个 JFrame 中 (本 书 所 有 的 示例 都 是 如 此 )。 静 态 的 run0 方 法 可 以 将 
视窗 的 标题 设置 为 类 的 简单 名 。 
练习 3: (3) 修改 SubmitSwingProgram.java， 让 它 使 用 SwingConsole。 


22.3 创建 按钮 


创建 一 个 按钮 非常 简单 : 只 要 用 你 希望 出 现在 按钮 上 的 标签 调用 JButton 的 构造 器 即 可 。 在 
后 面 你 会 看 到 一 些 更 有 趣 的 功能 ， 比 如 在 按钮 上 显示 图 形 。 

一 般 来 说 ， 要 在 类 中 为 按钮 创建 一 个 字段 ， 以 便 以 后 可 以 引用 这 个 按钮 。 

JButton 是 一 个 组 件 ， 它 有 自己 的 小 窗口 ， 能 作为 整个 更 新 过 程 的 一 部 分 而 自动 被 重 绘 。 也 
就 是 说 ， 你 不 必 显 式 绘制 一 个 按钮 或 者 别 的 类 型 的 控件 ， 只 要 把 它们 放 在 窗 体 上 ， 它 们 可 以 自 
动 绘制 自己 。 通 常 你 会 在 构造 器 内 部 把 按钮 加 入 窗 体 : 


/1/: gui/Button1.java 
// Putting buttons on a Swing application. 
import javax.swin, i 
import java.awt.*: 
import static net.mindview.util.SwingConsole.*; 





public class Buttonl extends JFrame { 
private JButton 
bl = new JButton("Button 1"), 
b2 = new JButton("Button 2"); 
public Button1() { 
setLayout (new FlowLayout()); 
add(b1) ; 
add(b2) ; 


} 
public static void main(String[] args) { 
run(new Buttoni(), 208, 100); 


F 
yh 
这 里 引入 了 一 些 新 内 容 : 在 向 JFrame 添 加 任何 组 件 之 前 ， 先 给 出 一 个 新 的 FlowLayout 类 型 
的 “布局 管理 器 ”"。 布 局 管理 器 是 面板 用 来 隐 式 地 决定 控件 在 窗 体 上 的 位 置 的 工具 。JFrame 通 党 
使 用 BorderLayout 管 理 布局 ， 但 这 里 不 能 使 用 (在 本 章 后 面部 分 将 学 习 它 )， 因 为 它 的 默认 行为 
是 每 加 入 一 个 控件 ， 将 完全 覆盖 其 他 控件 。FlowLayout 使 得 控件 可 以 在 窗 体 上 从 左 到 右 、 从 上 
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到 下 连续 均匀 分 布 。 
练习 4: (1) 验证 如 果 在 Button1.java 没 有 setLayout0 调 用 ， 那 么 就 只 有 一 个 按钮 会 出 现在 所 
产生 的 程序 中 。 


22.4 捕获 事件 


如 果 编 译 并 运行 前 面 的 程序 ， 那 么 当 按 下 按钮 的 时 候 ， 什 么 也 不 会 发 生 。 现 在 是 必须 深入 
进去 编写 一 些 代 码 以 决定 会 发 生 什么 事情 的 时 候 了 。 事 件 驱 动 编程 (包含 了 许多 关于 GUI 的 内 
容 ) 的 基础 ， 就 是 把 事件 同 处 理事 件 的 代码 连接 起 来 。 

在 Swing 中 ， 这 种 关联 的 方式 就 是 通过 清楚 地 分 离 接口 (图 形 组 件 ) 和 实现 〈 当 和 组 件 相关 
的 事件 发 生 时， 你 要 执行 的 代码 ) 而 做 到 的 。 每 个 Swing 组 件 都 能 够 报告 其 上 所 有 可 能 发 生 的 事 
件 ， 并 且 它 能 单独 报告 每 种 事件 。 所 以 、 你 要 是 对 诸如 “鼠标 移动 到 按钮 上 ”这 样 的 事件 不 感 
兴趣 的 话 ， 那 么 你 不 注册 这 样 的 事件 就 可 以 了 。 这 种 处 理事 件 驱 动 编程 的 方式 非常 直接 和 优雅 ， 
一 且 你 理解 了 其 基本 概念 ， 就 能 够 很 容易 将 其 应 用 到 甚至 从 未 见 过 的 Swing 组 件 之 上 。 实 际 上 ， 
只 要 是 JavaBean (本 章 后 面 讨论 ) ， 这 个 模式 都 适用 。 

首先 ， 对 所 使 用 的 组 件 ， 我 们 只 把 重点 放 在 它 感 兴趣 的 主要 事件 上 。 对 于 JButton,“ 感 兴趣 
的 事件 ”就 是 按钮 被 按 下 。 为 了 表明 (注册 ) 你 对 按钮 按 下 事件 感 兴趣 ， 可 以 调用 JButton 的 
addActionListener0) 方 法 。 这 个 方法 接受 一 个 实现 ActionListener 接 口 的 对 象 作 为 参数 ， 
ActionListener 接 口 只 包含 一 个 actionPerformed0 方 法 。 所 以 要 想 把 事件 处 理 代码 和 JButton 关 联 ， 
需要 在 一 个 类 中 实现 ActionListener 接 口 ， 然 后 把 这 个 类 的 对 象 通过 addActionListener0 方 法 注册 
给 JButton。 这 样 按钮 按 下 的 时 候 就 会 调用 actionPerformed( 方 法 (通常 这 也 称 为 回调 )。 

但 是 按钮 按 下 的 时 候 应 该 有 什么 结果 呢 ? 我 们 希望 看 到 屏幕 有 所 改变 ， 所 以 在 这 里 介绍 一 
个 新 的 Swing 组 件 一 JTextField。 这 个 组 件 支持 用 户 输入 文本 ， 在 本 例 中 ， 或 者 像 本 例 一 样 由 
程序 插入 文本 。 尽 管 有 很 多 方法 可 以 创建 JTextField， 但 是 最 简单 的 方式 就 是 告诉 构造 器 你 所 希 
望 的 文本 域 宽度 。 一 旦 JTextField 被 放置 到 窗 体 上 ， 就 可 以 使 用 setText0 方 法 来 修改 它 的 内 容 
(JTextField 还 有 很 多 方法 ， 不 过 你 应 该 先 到 java.sun.com 看 一 下 HTML 格 式 的 JDK 文 档 ) 。 下 面 就 
是 其 具体 程序 : 


/1: gui/Button2. java 
// Responding to button presses. 
import javax.swing.*; 
import java.awt.*; 
* import java.awt.event.*; 
‘import static net.mindview.util. SwingConsole.*; 


public class Button2 extends JFrame { 
private JButton 
bl = new JButton("Button 1"), 
b2 = new JButton("Button 2"); 
private JTextField txt = new JTextField(10) ; 
class ButtonListener implements ActionListener { 
public void actionPerformed(ActionEvent e) { 
String name = ((JButton)e.getSource()).getText (); 
txt. setText (name) ; 
} 
} 
private ButtonListener bl = new ButtonListener(); 
public Button2() { 
bl.addActionListener (b1); 
b2.addActionListener (bl); 
SetLayout (new FlowLayout()); 
add(b1); 
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add(b2) ; 
add(txt); 


) 
public static void main(String[] args) { 
run(new Button2(), 268, 150); 


) 
Mim 
创建 JTextField 并 把 它 放置 在 画布 上 的 步 又 ， 同 JButton 或 者 其 他 Swing 组 件 所 采用 的 步 又 相 
同 。 这 里 与 前 面 例子 的 不 同 之 处 在 于 创建 一 个 ButtonListener 对 象 ， 它 实现 了 前 面 提 到 过 的 
1313] ActionListener 接 口 。actionPerformed0 方 法 的 参数 是 ActionEvent 类 型 ， 它 包含 事件 和 事件 源 的 
所 有 信息 。 本 例 中 ， 我 希望 表明 是 哪个 按钮 被 按 下 ，getSource() 方 法 产生 的 对 象 表明 了 事件 的 
来 源 ， 我 假设 (使 用 类 型 转换 ) 这 个 对 象 是 JButton。getText0 方 法 返回 按钮 上 的 文本 ， 这 个 文 
本 被 放 进 JTextField， 以 证 明 当 按钮 按 下 的 时 候 代 码 确 实 被 调用 了 。 
在 构造 器 中 ， 使 用 addActionListener0 方 法 来 将 BuftonListener 对 象 注册 给 两 个 按钮 。 
通常 ， 把 ActionListener 实 现成 匿名 内 部 类 会 更 方便 ， 尤 其 是 对 每 个 监听 器 类 只 使 用 一 个 实 
例 的 时 候 更 是 如 此 。 可 以 像 下 面 这 样 修改 Button2.java， 这 里 使 用 一 个 匿名 内 部 类 : 





/1: gui/Button2b. java 

7/ Using anonymous inner classes. 

import javax.swing.*; 

import java.awt.*; 

import java.awt.event.*; 

import static net.mindview.util.SwingConsole.*; 





public class Button2b extends JFrame { 
private JButton 
bl = new JButton("Button 1"), 
b2 = new JButton("Button 2"); 
private JTextField txt = new JTextField(1@); 
private ActionListener bl = new ActionListener() { 
public void actionPerformed(ActionEvent e) { 
String name = ((JButton)e.getSource()).getText(); 
txt.setText (name) ; 
} 


X 

public Button2b() { 
bl1.addActionListener (bl) ; 
b2.addActionListener (bl) ; 
setLayout (new FlowLayout()) : 
add(b1); 
add(b2) ; 
add (txt); 


} 
public static void main(String(} args) { 
run(new Button2b(), 200, 156); 





ma 
dh 
本 书 中 的 例子 倾向 于 (只 要 可 能 ) 使 用 匿名 内 部 类 的 方式 。 
练习 5: (4) 使 用 SwingConsole 类 编写 一 个 应 用 程序 ， 它 包括 一 个 文本 域 和 三 个 按钮 ， 单 击 
每 个 按钮 的 时 候 ， 在 文本 域 中 显示 不 同 的 文字 。 


22.5 文本 区 域 


i 除了 可 以 有 多 行文 本 以 及 更 多 的 功能 不 同 之 外 ，JTextArea 与 JTextField 在 其 他 方面 都 很 相 
似 。JTextArea 有 一 个 比较 常用 的 方法 是 append0。 因 为 可 以 往 回 滚动 ， 所 以 比 起 在 命令 行程 序 
中 把 文本 打印 到 标准 输出 的 做 法 ， 这 就 成 为 了 一 种 进步 。 例 如 ， 下 面 的 程序 使 用 第 17 章 中 的 
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Countries 生 成 器 的 输出 来 填充 JTextArea。 


711: gui/TextArea. java 
// Using the JTextArea control. 

import javax.swing.*; 

import java.awt.*; 

import java.awt.event.*; 

import java.util. *; 

import net.mindview.util.*; 

import static net.mindview.util.SwingConsole.*; 


public class TextArea extends JFrame { 
private JButton 
b = new JButton("Add Data"), 
¢ = new JButton("Clear Data"): 
private JTextArea t = new JTextArea(20, 49); 
private Map<String, String> m = 
new HashMap<String, String>() : 
public TextArea() { 
// Use up all the data: 
m.putALl (Countries.capitals()): 
b.addActionListener(new ActionListener() { 
public void actionPerformed(ActionEvent e) { 
for(Map.Entry me : m.entrySet()) 
t.append(me.getKey() + ": "+ me.getValue()+"\n"); 
} 
Ds 1315} 
c.addActionListener(new ActionListener() { 
public void actionPerformed(ActionEvent e) { 
t.setText(""); 














H; 

setLayout(new FlowLayout()); 
add(new JScrollPane(t)); 

add (b) ; 

add(c); 


} 
public static void main(Striñg[] args) { 
run(new TextArea(), 475, 425); 


} 

Mix 

在 构造 器 中 ， 用 国家 及 其 首都 名 称 来 填充 Map。 注 意 ， 对 于 其 中 的 两 个 按钮 ， 因 为 在 程序 
中 你 不 再 需要 引用 监听 器 ， 所 以 直接 创建 ActionListener 对 象 并 添加 ， 而 没有 定义 中 间 变量 。 
“Add Data” 按 钮 格式 化 并 添加 所 有 数据 ，“Clear Data” 按 钮 使 用 setText0 方 法 来 清理 JTextArea 
中 的 所 有 文本 。 

在 JTextArea 被 添加 到 JErame 中 之 前 ， 先 被 包装 进 了 JSerollPane ( 当 屏幕 上 的 文本 太 多 的 
时 候 用 它 来 进行 滚动 控制 )。 这 么 做 就 足以 得 到 完整 的 滚动 功能 。 由 于 我 曾 试图 在 其 他 GUI 编程 
环境 中 得 到 类 似 功能 ， 所 以 我 对 像 JSerollPane 这 样 设计 良好 、 使 用 简单 的 组 件 印象 非常 深刻 。 

练习 6: (7) 将 strings/TestRegularExpression.java 转 变 为 可 交互 的 Swing 程 序 ， 使 得 你 可 以 在 
一 个 JTextArea 中 放置 输入 字符 串 ， 在 另 一 个 JTextField 中 放置 正则 表达 式 。 运 行 结果 应 该 在 第 
二 个 JTextArea 中 显示 。 

练习 7: (5) 使 用 SwingConsole 编 写 一 个 应 用 程序 ， 添 加 所 有 具有 addActionListener0 方 法 的 
Swing 组 件 (在 http://java.sun.com 的 JDK 文 档 中 查找 这 些 组 件 。 提 示 : 使 用 索引 功能 搜索 
addActionListener0)。 针 对 每 个 组 件 ， 捕 获 其 事件 ， 并 在 文本 域 中 显示 相应 的 信息 。 

练习 8: (6) 几乎 所 有 的 Swing 组 件 都 是 从 Component 导 出 的 ，Component 有 一 个 setCursor0 方 
法 。 在 JDK 文 档 中 查找 有 关内 容 。 创 建 一 个 应 用 程序 ， 将 光标 修改 为 Carsor 类 中 存储 的 光标 之 一 。 
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22.6 控制 布局 


在 Java 中 ,组 件 放 置 在 窗 体 上 的 方式 可 能 与 你 使 用 过 的 任何 GUI 系统 都 不 相同 。 首 先 ， 它 完 
全 基于 代码 ， 没 有 用 来 控制 组 件 布置 的 “资源 "。 第 二 ， 组 件 放置 在 窗 体 上 的 方式 不 是 通过 绝对 
坐标 控制 ， 而 是 由 “布局 管理 器 ”根据 组 件 加 入 的 顺序 决定 其 位 置 。 使 用 不 同 的 布局 管理 器 ， 
组 件 的 大 小 、 形 状 和 位 置 将 大 不 相同 。 此 外 ， 布 局 管理 器 还 可 以 适应 applet 或 应 用 程序 窗口 的 大 
小 ， 所 以 如 果 窗口 的 尺寸 改变 了 ， 组 件 的 大 小 、 形 状 和 位 置 也 能 够 做 相应 的 改变 。 

JApplet、JFrame、JDialog、JPanel 等 都 可 以 包含 和 显示 组 件 。Container 中 有 一 个 称 为 
setLayout() 的 方法 ， 可 以 通过 这 个 方法 来 选择 不 同 的 布局 管理 器 。 在 本 节 中 ， 我 们 将 通过 在 窗 
体 上 放置 一 些 按钮 来 研究 不 同 的 布局 管理 器 (这 样 最 简单 )。 这 些 示例 不 会 捕获 任何 按钮 事件 ， 
因为 它们 仅仅 是 为 了 演示 按钮 是 如 何 布局 的 。 
22.6.1 BorderLayout 

除非 你 设置 为 其 他 的 布局 模式 ， 否 则 JFrame 将 使 用 BoarderLayout 作 为 默认 的 布局 模式 。 
如 果 不 加 入 其 他 指令 ， 它 将 接受 你 调用 add0 方 法 而 加 入 的 组 件 ， 把 它 放置 在 中 央 ， 然 后 把 组 件 
向 各 个 方向 拉 伸 ， 直 到 与 边框 对 齐 。 

BorderLayout 具 有 四 个 边框 区 域 和 一 个 中 央 区 域 的 概念 。 当 向 由 BorderLayout 管 理 的 面板 
加 入 组 件 的 时 候 ， 可 以 使 用 重 载 的 add(0 方 法 ， 它 的 第 一 个 参数 接受 一 个 常量 值 。 这 个 值 可 以 为 
以 下 任何 一 个 ; 





BorderLayout NORTH 顶端 
BorderLayout. SOUTH 底 端 
BorderLayout. EAST Hii 
BorderLayout. WEST Zi 
BorderLayout.CENTER 从 中 央 开始 填充 ， 直 到 与 其 他 组 件 或 边框 相遇 


如 果 没 有 为 组 件 指定 放置 的 位 置 ， 默 认 情况 下 它 将 被 放置 到 中 央 。 
在 下 面 的 示例 中 使 用 了 默认 布局 ， 因 为 默认 情况 下 JEFrame 使 用 的 就 是 BorderLayout; 


/1: gui/BorderLayout1.java 
// Demonstrates BorderLayout . 

import javax. swing.*; 

import java.awt.*; 

import static net.mindview.util.SwingConsole..*; 


public class BorderLayout1 extends JFrame { 
public BorderLayoutl() { 
add(BorderLayout .NORTH, new JButton("North")); 
add (BorderLayout.SOUTH, new JButton("South")); 
add (BorderLayout.EAST, new JButton("East")); 
add(BorderLayout.WEST, new JButton("West")): 
add(BorderLayout CENTER, new JButton("Center")); 


} 
public static void main(String(] args) { 
run(new BorderLayouti1(), 366, 250); 


} 
bh 
对 于 除 CENTERI 以 外 的 所 有 位 置 ， 加 入 的 组 件 将 被 沿 着 一 个 方向 压缩 到 最 小 尺寸 ， 同 时 在 另 一 
个 方向 上 拉 伸 到 最 大 尺寸 。 不 过 对 于 CENTER， 组 件 将 在 两 个 方向 上 同时 拉 伸 ， 以 覆盖 中 央 区 域 。 
22.6.2 FlowLayout 
它 直 接 将 组 件 从 左 到 右 “ 流 动 ”到 窗 体 上 ， 直 到 占 满 上 方 的 空间 ， 然 后 向 下 移动 一 行 ， 继 
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续 流动 。 

在 下 面 的 例子 中 ， 先 把 布局 管理 器 设置 为 FlowLayout， 然 后 在 窗 体 上 放置 按钮 。 你 将 注意 
到 ， 在 使 用 FlowLayout 的 情况 下 ， 组 件 将 呈现 出 “合适 ”的 大 小 。 比 如 ， 一 个 JButton 的 大 小 就 
是 其 标签 的 大 小 。 


//: gui/FlowLayout1. java 

// Demonstrates FlowLayout. 

import javax.swing.*; 

import java.awt.*; 

import static net.mindview.util.SwingConsole.*; 


public class FlowLayout1 extends JFrame { 
public FlowLayoutl() { 
setLayout(new FlowLayout()); 
for(int 1 = 0; 1 < 20; i++) 
add(new JButton("Button ”+ 1)); 


} 
public static void main(String{] args) { 
run(new FlowLayout(), 360, 300): 


} 
dh 


使 用 FlowLayout， 所 有 的 组 件 将 被 压缩 到 它们 的 最 小 尺寸 ， 所 以 可 能 会 得 到 令 人 惊讶 的 效 
果 。 比 如 ， 在 使 用 FlowLayout 的 时 候 ， 因 为 JLabel 的 尺寸 就 是 其 字符 串 的 尺寸 ， 这 就 使 得 文本 
右 对 齐 不 会 产生 任何 视觉 上 的 效果 。 

请 注意 : 如 果 你 调整 视窗 的 尺寸 ， 那 么 布局 管理 器 将 随 之 重新 流动 所 有 组 件 。 

22.6.3 GridLayout 

GridLayout 人 允许 你 构建 一 个 放置 组 件 的 表格 ， 在 向 表格 里 面 添加 组 件 的 时 候 ， 它 们 将 按照 
从 左 到 右 、 从 上 到 下 的 顺序 加 入 。 在 构造 器 中 要 指定 需要 的 行 数 和 列 数 ， 它 们 将 均匀 分 布 在 窗 
体 上 。 

11: gut/GridLayout1.java 

// Demonstrates GridLayout. 

import javax.swing.*; 


import java.awt.*; 
import static net.mindview.util.SwingConsole.*; 


public class GridLayout1 extends JFrame { 
public GridLayoutl() { 
setLayout(new GridLayout(7.3)): 
for(int i = 9; i < 20; i++) 
add(new JButton("Button " + 1)); 


} 
public static void main(String[] args) { 
run(new GridLayout1(), 300, 300); 


} Yi 

在 这 个 例子 中 有 21 个 空位 ， 但 是 只 加 入 了 20 个 按钮 。 因 为 GridLayout 并 不 进行 “均衡 ”处 
理 ， 所 以 最 后 一 个 空位 将 被 闲置 。 
22.6.4 GridBagLayout 

GridBagLayout 提 供 了 强大 的 控制 功能 ， 包 括 精确 判断 视窗 区 域 如 何 布 局 ， 以 及 视窗 大 小 
变化 的 时 候 如 何 重新 放置 组 件 。 不 过 ， 它 也 是 最 复杂 的 布局 管理 器 ， 所 以 很 难 理解 。 它 的 目的 
主要 是 辅助 GUI 构造 工具 〈 它 可 能 使 用 GridBagLayout 而 不 是 绝对 位 置 来 控制 布局 ) 自动 生成 代 
码 。 如 果 你 发 现 自己 的 设计 非常 复杂 ， 以 至 于 需要 使 用 GridBagLayout， 那 么 你 应 该 使 用 GUI 构 
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造 工具 来 生成 这 个 设计 。 如 果 读者 觉得 自己 必须 掌握 它 的 复杂 细节 ， 我 推荐 读者 参考 专门 的 
Swing 书 作为 起 点 。 

作为 一 种 可 替换 的 选择 ， 你 可 能 会 考虑 TableLayout， 它 不 属于 Swing 类 库 ， 但 是 可 以 从 
http://java.sun.com 处 下 载 。 这 个 组 件 被 置 于 GridBagLayout 之 上 ， 并 且 隐 藏 了 其 大 多 数 细节 ， 
此 可 以 极 大 地 简化 使 用 这 种 模式 的 方式 。 

22.6.5 绝对 定位 

我 们 也 可 以 设置 图 形 组 件 的 绝对 位 置 : 

1) 使 用 setLayout(nu 了 1 方法 把 容器 的 布局 管理 器 设置 为 空 。 

2) 为 每 个 组 件 调用 setBounds0 或 者 reshape0 方 法 (取决 于 语言 的 版 本 )， 为 方法 传递 以 像素 
坐标 为 单位 的 边界 矩形 的 参数 。 根 据 你 要 达到 的 目的 ， 可 以 在 构造 器 或 者 paint() 方 法 中 调用 这 
些 方法 。 

某 些 GUI 构 造 工 具 大 量 使 用 这 种 方法 ， 不 过 这 通常 不 是 生成 代码 的 最 佳 方式 。 

22.6.6 BoxLayout 

由 于 淡 们 在 理解 和 使 用 GridBagLayout 的 时 候 遇 到 了 很 多 问题 ， 所 以 Swing 还 提供 了 
BoxLayout， 它 具有 GridBagLayout 的 许多 好 处 ， 却 不 像 GridBagLayout 那 么 复杂 。 所 以 当 你 需 
要 手工 编写 布局 代码 的 时 候 ， 可 以 考虑 使 用 它 ( 再 次 提醒 读者 ， 如 果 你 的 设计 过 于 复杂 ， 那 么 
就 应 该 使 用 GUI 构 造 工具 来 生成 布局 代码 )。BoxLayout 使 你 可 以 在 水 平方 向 或 者 垂直 方向 控制 
组 件 的 位 置 ， 并 且 通 过 所 谓 的 “支架 和 胶水 ”(struts and glue) 的 机 制 来 控制 组 件 的 间隔 。 你 可 
以 在 www.MindView 上 的 本 书 在 线 补充 材料 中 找到 若干 使 用 BoxLayout 的 基本 示例 。 

22.6.7 最 好 的 方式 是 什么 

Swing 功能 强大 ， 用 少数 几 行 代码 就 可 以 做 很 多 事情 。 基 于 学 习 的 目的 ， 本 书 中 的 例子 相当 
简单 ， 所 以 手工 编写 它们 很 有 意义 。 通 过 组 合 简单 布局 ， 就 能 得 到 非常 多 的 结果 。 不 过 ， 在 某 
些 情况 下 ， 手 工 编写 GUI 窗 体 就 不 太 适 合 了 ， 这 样 做 太 复杂 ， 也 不 能 充分 利用 编程 时 间 。Java 和 
Swing 设 计 者 的 最 初 目 的 就 是 要 使 语言 和 库 能 对 GUI 构 造 工具 提供 支持 ， 创 建 这 些 工具 的 明确 的 
目的 也 是 为 了 使 你 更 容易 地 获取 编程 经 验 。 只 要 理解 了 布局 的 方式 以 及 如 何 处 理事 件 ( 下 面 将 
学 习 到 )， 那 么 如 何 手工 放置 组 件 的 细节 就 显得 不 那么 重要 了 ， 应 该 让 合适 的 工具 帮 你 去 做 这 些 
事情 〈 毕 竞 ， 设 计 Java 的 目的 是 为 了 提高 程序 员 的 生产 率 ) 。 


22.7 Swing 事件 模型 


在 Swing 的 事件 模型 中 ,组 件 可 以 发 起 (触发 ) 一 个 事件 。 每 种 事件 的 类 型 由 不 同 的 类 表示 。 
当 事 件 被 触发 时 ， 它 将 被 一 个 或 多 个 “监听 器 ”接收 ， 监 听 器 负责 处 理事 件 。 所 以 ， 事 件 发 生 
的 地 方 可 以 与 事件 处 理 的 地 方 分 离开 。 既 然 是 以 这 种 方式 使 用 Swing 组 件 ， 那 么 就 只 需 编写 组 件 
收 到 事件 时 将 被 调用 的 代码 ， 所 以 这 是 一 个 分 离 接口 与 实现 的 极 佳 例子 。 

所 谓 事 件 监听 器 ， 就 是 一 个 “实现 特定 类 型 的 监听 器 接口 ”的 类 对 象 。 所 以 程序 员 要 做 的 
就 是 ， 先 创建 一 个 监听 器 对 象 ， 然 后 把 它 注册 到 触发 事件 的 组 件 。 这 个 注册 动作 是 通过 调用 触 
发 事件 的 组 件 的 addXXXListener0 方 法 来 完成 的 ， 这 里 用 XXX 表示 监听 器 所 监听 的 事件 类 型 。 
通过 观察 addListener 方 法 的 名 称 ， 就 可 以 很 容易 地 知道 其 能 够 处 理 的 事件 类 型 ， 要 是 你 把 所 监听 
事件 的 类 型 搞 错 了 ， 在 编译 期 间 就 会 发 现 有 错误 。 在 本 章 的 后 面 将 会 学 习 到 ，JavaBean 也 是 使 用 
addListener 方 法 名 称 来 判断 某 个 Bean 所 能 处 理 的 事件 类 型 的 。 

然后 ， 所 有 的 事件 处 理 逻 辑 都 将 被 置 于 监听 器 类 的 内 部 。 要 编写 一 个 监听 器 类 ， 唯 一 的 要 
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求 就 是 必须 实现 相应 的 接口 。 可 以 创建 一 个 全 局 的 监听 器 类 ， 不 过 有 了 时 写成 内 部 类 会 更 有 用 。 
这 不 仅 是 因为 将 监听 器 类 放 在 它们 所 服务 的 用 户 接口 类 或 者 业务 逻辑 类 的 内 部 时 ， 可 以 在 逻辑 
上 对 其 进行 分 组 ， 而 且 还 因为 (将 在 后 面 看 到 ) 内 部 类 对 象 含有 一 个 对 其 外 部 类 对 象 的 引用 ， 
这 就 为 跨越 类 和 子 系统 边界 的 调用 提供 了 一 种 优雅 的 方式 。 

到 目前 为 止 ， 在 本 章 的 所 有 例子 中 已 经 使 用 了 Swing 事件 模型 ， 本 节余 下 部 分 将 补充 这 个 模 
型 的 细节 。 


22.7.1 事件 与 监听 器 的 类 型 

所 有 Swing 组 件 都 具有 addXXXListener0 和 removeXXXListener( 方 法 。 这 样 就 可 以 为 每 个 
组 件 添加 或 移 除 相应 类 型 的 监听 器 。 注 意 ， 每 个 方法 的 “XXX” 还 表示 方法 所 能 接收 的 参数 ， 
比如 addMyListener(MyListenerm)。 下 表 包 含 相互 关联 的 基本 事件 、 监 听 器 以 及 通过 提供 
addXXXListener0 和 removeXXXListener( 方 法 来 支持 这 些 事件 的 基本 组 件 。 记 住 ， 事 件 模型 是 
可 以 扩展 的 ， 所 以 将 来 你 也 许 会 遇 到 表格 里 没有 列 出 的 事件 和 监听 器 。 
—— -~ 

事件 、 监 听 器 接口 以 及 “添加 ” 

和 “ 移 除 ”方法 支持 此 事件 的 组 件 

ActionEvent JButton、JList、JTextField、JMenultem 及 其 派生 类 ， 包 括 JCheckBox- 

ActionListener Menultem, JMenufnJRadioButtonMenu Item 

addActionListener() 

removeActionListener() 

AdjustmentEvent JSerolbar 以 及 你 编写 的 任何 实现 Adjustable 接 口 的 类 

AdjustmentListener 

addAdjustmentListener() 

removeAdjustmentListener() 

ComponentEvent “Component 及 其 派生 类 ， 包 括 JButton, JCheckBox, JComboBox, 

ComponentListener Container, JPanel, JApplet, JScrollPane, Window, JDialog, 

addComponentListener() JFileDialog, JFrame, JLabel, JList, JScrolibar, JTextAreaffiJTextField 

removeComponentListener() 

ContainerEvent Container 及 其 派生 类 ， 包 括 JSerollPane、Window、JDialog、JFileDialog 

addContainerListener0) 和 JFrame 

removeContainerListener() 

FocusEvent Component 及 其 派生 类 * 

FocusListener 

addFocusListener() 

removeFocusListener() 

KeyEvent Component 及 其 派生 类 * 

KeyListener 

addKeyListener() 

removeKeyListener() 

MouseEvent (包括 单 击 和 移动 ) Component 及 其 派生 类 * 

MouseListener 

addMouseListener0 

removeMouseListener0 

MouseEvent® (包括 单 击 和 移动 ) Component 及 其 派生 关 * 








O 尽管 表示 鼠标 移动 的 事件 似乎 很 有 必要 ， 但 Swing 并 没有 提供 NiouseMotionEvent 这 样 的 事件 。MouseEvent 包 
含 了 鼠标 单 击 和 移动 的 事件 ， 所 以 MouseEvent 在 这 个 表 中 第 二 次 出 现 并 不 是 一 个 错误 。 
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事件 、 监 听 器 接口 以 及 “添加 ” 

和 “ 移 除 ”方法 支持 此 事件 的 组 件 
MouseMotionListener 

addMouseMotionListener() 

removeMouseMotionListenerO 

WindowEvent Window 及 其 派生 类 ， 包 括 JDialog、JFileDialog 和 JFrame 
WindowListener 

addWindowListener0 

removeWindowListener() 

TtemEvent JCheckBox、JCheckBoxMenultem、JComboBox、JList 以 及 任何 实现 
ltemListener 了 ItemSelectable 接口 的 类 

additemListener() 

removeltemListener() 

TextEvent 任何 从 JTextComponent 导 出 的 类 ， 包 括 JTextArea 和 JTextField 
TextListener 

addTextListener() 

removeTextListener() 


读者 可 以 观察 到 ， 每 种 组 件 所 支持 的 事件 类 型 都 是 固定 的 。 为 每 个 组 件 列 出 其 支持 的 所 有 
事件 是 相当 困难 的 。 一 个 比较 简单 的 方法 是 修改 第 14 章 的 ShowMethods.java 程 序 ， 这 样 它 就 可 
以 显示 出 你 所 输入 的 任意 Swing 组 件 所 支持 的 所 有 事件 监听 器 。 

第 14 章 介绍 了 反射 机 制 ， 并 且 使 用 反射 对 指定 的 类 查找 其 方法 : 既 可 以 查找 所 有 方法 的 列 
表 ， 也 可 以 查找 “方法 名 称 符合 你 所 提供 的 关键 字 的 ”部 分 方法 。 反 射 的 神奇 之 处 在 于 它 能 自 
动 得 到 一 个 类 的 所 有 方法 ， 而 不 用 遍历 类 的 整个 继承 层次 并 在 每 个 层次 检查 基 类 。 所 以 ， 它 为 
编程 提供 了 极 具 价值 并 可 以 节省 时 间 的 工具 ， 因 为 大 多 数 Java 方 法 的 名 称 非常 详细 且 具 有 描述 
性 ， 所 以 可 以 找 出 包含 你 所 感 兴趣 的 关键 字 的 方法 名 称 。 当 找到 了 所 要 找 的 方法 后 ， 就 可 以 在 
JDK 文 档 里 查看 其 细节 了 。 

下 面 是 ShowMethods.java 的 更 好 用 的 GUI 版 本 ， 专 门 用 来 查找 Swing 组 件 里 的 addListener 
方法 : 


//: gui/ShowAddListeners.java 

// Display the “addXxXListener” methods of any Swing class. 
import javax.swing.*: 
import java.awt.*; 
import java.awt.event.*; 
import java. lang. ref 
import java.util.regex. 
import static net.mindview.util.SwingConsole.*; 











public class ShowAddListeners extends JFrame { 
private JTextField name = new JTextField(25); 
private JTextArea results = new JTextArea(40, 65); 
private static Pattern addListener = 
Pattern. compile("(add\\w+?Listener\\(.*?\\))"); 
private static Pattern qualifier = 
Pattern.compile("\\w+\\."): 
class Namel implements ActionListener { 
public void actionPerformed(ActionEvent e) { 
String nm = name.getText().trim(): 
if(nm.length() == @) { 
Fesults.setText("No match"); 
return; 


| 
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F 
Class<?> kind: 
try { 
kind = Class. forName("javax.swing.“ + nm); 
} catch(ClassNotFoundException ex) { 
results.setText("No match"): 
return: 


} 
Method[] methods = kind.getMethods(); 
résults.setText(""); 
for (Method m : methods) { 
Matcher matcher = 
addListener matcher (m. toString()); 
if (matcher. find()) 
results. append (qualifier .matcher( 
matcher .group(1)).replaceAll("") + "\n"): 
} 
} 


} 

public ShowAddListeners() { 
‘NameL nameListener = new NameL(); 
name. addAct ionL i stener (nameListener) ; 
JPanel top = new JPanel(); 
top.add(new JLabel("Swing class name (press Enter):")); 
top. add (name) ; 
add(BorderLayout .NORTH, top): 
add(new JScrollPane(results)) ; 
// Initial data and test: 
name. setText ("JTextArea") ; 
nameListener . act ionPerformed( 

new ActionEvent("", © ."")); 


} 
public static void main(String{} args) { 
run(new ShowAddListeners(), 500, 408); 


} 

} M~ 

在 name JTextField 中 输入 要 查找 的 Swing 组 件 类 的 名 称 。 查 找 的 结果 将 使 用 正则 表达 式 进行 
匹配 ， 最 终结 果 显 示 在 JTextArea 中 。 

注意 ， 这 里 没有 使 用 按钮 或 者 别 的 组 件 来 表明 你 希望 启动 查找 。 这 是 由 于 JTextField 被 
ActionListener 所 监听 。 当 你 做 出 更 改 并 按 下 “ 回 车 ” 键 后 ， 列 表 马 上 就 得 到 了 更 新 。 如 果 文本 
域 的 内 容 非 空 ， 将 把 此 内 容 作为 Class.forName0 的 参数 ， 以 用 来 查找 这 个 类 。 如 果 名 称 不 正确 ， 
Class.forName() 方 法 将 失败 ， 即 抛 出 异常 。 这 个 异常 将 被 捕获 ， 并 把 JTextArea 内 容 设置 为 No 
match (不 匹配 ) 。 如 果 输 入 正确 的 名 称 〈 注 意 大 小 写 ) ，Class.forName() 将 成 功 返回 ， 然 后 
getMethods() 方 法 将 返回 一 个 Method 对 象 的 数组 。 

这 里 使 用 了 两 个 正则 表达 式 。 第 一 个 是 addListener， 它 查找 的 模式 为 ， 以 add 开 头 ， 后 面 跟 
任意 字母 ， 然 后 接 Listener， 最 后 是 括号 内 的 参数 列表 。 注 意 ， 整 个 正则 表达 式 用 “ 非 转 义 ”的 括 
号 包围 ， 意 思 是 当 发 生 匹 配 的 时 候 ， 它 可 以 作为 一 个 正则 表达 式 “ 组 ”来 访问 。 在 
NameL.ActionPerformed0 中 ， 通 过 把 每 个 Method 对 象 都 以 字符 串 形式 传递 给 Pattern.matcher0 方 
法 ， 创 建 一 个 Matcher 对 象 。 当 在 此 对 象 上 调用 find0 的 时 候 ， 只 有 发 生 了 匹配 ， 才 会 返回 真 ， 这 
时 ， 你 可 以 通过 调用 group(D 来 选择 第 一 个 匹配 的 包含 在 括号 中 的 表达 式 组 。 这 样 得 到 的 字符 捉 
仍然 包含 限定 词 ， 为 了 把 限定 词 别 除 掉 ， 需 要 使 用 gualifier Pattern 对 象 ， 这 与 ShowMethodsjava 
中 的 做 法 很 相似 。 

在 构造 器 的 未 尾 ， 在 name 中 设置 一 个 初始 值 ， 然 后 触发 事件 ， 对 初始 数据 进行 一 次 测试 。 

这 个 程序 为 查询 Swing 组 件 所 支持 的 事件 类 型 提供 了 一 种 便利 方式 。 一 旦 知道 了 某 个 组 件 支 
持 哪些 事件 ， 不 用 参考 任何 资料 就 可 以 处 理 这 个 事件 了 。 你 只 要 : 
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D 获取 事件 类 的 名 称 ， 并 移 除 单词 “Event”， 然 后 将 剩 下 的 部 分 加 上 单词 “Listener”， 得 
到 的 就 是 内 部 类 必须 实现 的 监听 器 接口 。 

2) 实现 上 面 的 接口 ， 为 要 捕获 的 事件 编写 出 方法 。 比 如 ， 你 可 能 要 查找 鼠标 移动 ， 所 以 你 
可 以 为 MouseMotionListener 接 口 的 mouseMoved0 方 法 编写 代码 (自然 必须 同时 实现 接口 的 其 
他 方法 ， 不 过 很 快 你 会 学 到 一 种 简单 的 方式 )。 

3) 为 第 二 步 编写 的 监听 器 类 创建 一 个 对 象 。 然后 通过 调用 方法 向 组 件 注册 这 个 对 象 一 方 
法 名 为 “add” 前 级 加 上 监听 器 名 称 ， 比 如 addMouseMotionListener0。 

1326 下 面 是 一 些 监 听 器 接口 ， 











监听 器 接口 及 其 适配器 接口 中 的 方法 
ActionListener actionPerformed(ActionEvent) 
AdjustmentListener adjustmentValueChanged( 
AdjustmentEvent) 
ComponentListener ‘componentHidden(ComponentE vent) 
‘ComponentAdapter ‘componentShown(ComponentEvent) 


‘componentMoved(ComponentEvent) 
‘componentResized(ComponentE vent) 


ContainerListener componentAdded(ContainerEvent) 
ContainerAdapter componentRemoved(ContainerEvent) 
FocusListener focusGained(FocusE vent) 
FocusAdapter focusLost(FocusE vent) 
KeyListener keyPressed(KeyEvent) 
KeyAdapter keyReleased(KeyEvent) 
keyTyped(KeyEvent) 
MouseListener mouseClicked(MouseE vent) 
MouseAdapter mouseEntered(MouseEvent) 
mouseExited(MouseE vent) 
mousePressed(MouseEvent) 
mouseReleased(MouseE vent) 
MouseMotionListener mouseDragged(MouseE vent) 
MouseMotionAdapter mouseMoved(MouseE vent) 
WindowListener windowOpened(WindowEvent) 
WindowAdapter windowClosing(WindowEvent) 
windowClosed(WindowEvent) 
windowActivated(WindowE vent) 
windowDeactivated(WindowE vent) 
‘windowlconified(WindowE vent) 
windowDeiconified(WindowEvent) 
ItemListener itemStateChanged(ItemEvent) 





这 并 不 是 个 完整 的 列表 ， 部 分 原因 是 由 于 事件 模型 允许 你 编写 自己 的 事件 类 型 和 相应 的 监 
听 器 。 所 以 ， 人 们 常常 会 遇 到 含有 自 定义 事件 的 库 ， 本 章 学 习 到 的 知识 可 以 帮助 读者 理解 如 何 
[1327] 使 用 这 些 事件 。 
使 用 监听 器 适配器 来 进行 简化 
在 上 面 的 表 中 可 以 发 现 ， 某 些 监听 器 接口 只 有 一 个 方法 。 这 种 接口 实现 起 来 很 简单 。 不 过 ， 
具有 多 个 方法 的 监听 器 接口 使 用 起 来 却 不 太 方便 。 比 如 ， 如 果 你 想 捕获 一 个 鼠标 单 击 事件 (A 
如 ， 某 个 按钮 还 没有 替 你 捕获 该 事件 ) ， 那 么 就 需要 为 mouseClicked( 方 法 编写 代码 。 但 是 因为 
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MouseListener 是 一 个 接口 ， 所 以 尽管 接口 里 的 其 他 方法 对 你 来 说 没有 任何 用 处 ， 但 是 你 还 是 必 
须要 实现 所 有 这 些 方法 。 这 非常 烦人 。 

要 解决 这 个 问题 ， 某 些 (不 是 所 有 的 ) 含有 多 个 方法 的 监听 器 接口 提供 了 相应 的 适配器 
(可 以 在 上 面 的 表 中 看 到 具体 的 名 称 ) 。 适 配器 为 接口 里 的 每 个 方法 都 提供 了 默认 的 空 实现 。 现 
在 你 要 做 的 就 是 从 适配器 继承 ， 然 后 仅 覆 盖 那 些 需 要 修改 的 方法 。 比 如 ， 你 要 用 的 典型 的 
MouseListener 像 这 样 : 


class MyHouseListener extends HouseAdapter { 
public void mouseClicked(MouseEvent e) { 
71 Respond to mouse click... 
} 
+ 


适配器 的 出 发 点 就 是 为 了 使 编写 监听 器 类 变 得 更 容易 。 不 过 ， 适 配器 也 有 某 种 形式 的 缺陷 。 
假设 你 写 了 一 个 与 前 面 类 似 的 MouseAdapter: oo 

class MyMouseListener extends MouseAdapter { 

public void MouseClicked(MouseEvent e) { 
// Respond to mouse click... 

2 
这 个 适配器 将 不 起 作用 ， 而 且 要 想 找 出 问题 的 根源 也 非常 困难 ， 这 足以 让 你 发 疯 。 因 为 除了 鼠 
标 单 击 的 时 候 方 法 没有 被 调用 以 外 ， 程 序 的 编译 和 运行 都 十 分 良好 。 你 能 发 现 这 个 问题 吗 ? 它 
出 在 方法 的 名 称 上 : 这 里 的 名 称 是 MouseClicked0 而 没有 写成 mouseClicked0。 这 个 简单 的 大 小 
写 错误 导致 加 入 了 一 个 新 方法 。 它 不 是 关闭 视窗 的 时 候 所 应 该 调用 的 方法 ， 所 以 无 法 得 到 所 希 
望 的 结果 。 尽 管 使 用 接口 有 些 不 方便 ， 但 可 以 保证 方法 被 正确 实现 。 

要 想 保证 实际 上 的 确 是 覆盖 了 某 个 方法 ， 一 种 改进 的 方法 是 在 这 段 代码 的 上 面 使 用 内 建 的 
@Override 注 解 。 

练习 9: (5) 在 ShowAddListenersjava 的 基础 上 编写 程序 ， 实 现 typeinfo.ShowMethods java 
程序 的 完全 功能 。 

22.7.2 跟踪 多 个 事件 

作为 一 个 有 趣 的 试验 ， 也 为 了 向 读者 证 明 这 些 事件 确实 可 以 被 触发 ， 编 写 一 个 程序 ， 使 其 
能 够 跟踪 JButton 除 了 “是 否 被 按 下 ”事件 以 外 的 行为 ， 将 会 显得 很 有 价值 。 这 个 例子 还 向 读者 
演示 如 何 从 JButton 中 继承 出 自己 的 按钮 对 象 。 

在 下 面 的 代码 中 ，MyButton 是 TrackEvent 类 的 内 部 类 ， 所 以 MyButton 能 访问 父 窗口 ， 并 
操作 其 文本 区 域 ， 这 正 是 能 够 把 状态 信息 写 到 父 窗 体 的 文本 区 域内 所 必需 的 。 当 然 ， 这 是 一 个 
受 限 的 解决 方案 ， 因 为 MyButton 被 局 限于 只 能 与 TrackEvent 一 起 使 用 。 这 种 情况 有 时 称 为 “高 
耦合 ”代码 ; 


41: gui/TrackEvent. java 

// Show events as they happen. 
import javax.swin, 
import java.awt.*; 
import java.awt.event.*; 

import java.util.*; 

import static net.mindview.util. SwingConsole.*; 








public class TrackEvent extends JFrame { 
private HashMap<String, JTextField> h = 
new HashMap<String, JTextField>(): 


O 在 Java 1.0/1.1 版 中 ， 你 不 能 有 效 地 通过 继承 得 到 自己 的 按钮 对 象 。 这 只 是 其 基础 设计 中 的 众多 缺陷 之 一 。 
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private String[] event = { 


"focusLost", “keyPressed” 
"keyTyped". "mouseClicked 


“focusGained” 
"keyReleased 
"mouseEntered”, "mouseExited” 






private MyButton 


b1 = new MyButton(Color.BLUE, “test1"), 
b2 = new MyButton(Color.RED, “test2"); 


class MyButton extends JButton { 


void report (String field, String msg) { 
h.get (field). setText (msg); 
} 
FocusListener fl = new FocusListener() { 
public void focusGained(FocusEvent e) { 
report("focusGained", e.paramString()); 
} 
public void focusLost(FocusEvent e) { 
report("focusLost", e.paramString()); 





i 
KeyListener kl = new KeyListener() { 
public void keyPressed(KeyEvent e) { 
report("keyPressed", e.paramString()); 
} 


Public void keyReleased(KeyEvent e) { 
report("keyReleased", €.paramString()); 
$ 


public void keyTyped(KeyEvent e) { 
report("keyTyped", e.paramString()); 
t 
}; 
HouseListener ml = new MouseListener() { 
public void mouseClicked(MouseEvent e) { 
report(*mouseClicked", e.paramString()): 
: 
public void mouseEntered(MouseEvent e) { 
report (*mouseEntered", e.paramString()); 
} 
public void mouseExited(HouseEvent e) { 
report (*mouseExited", e.paramString()); 
} 
public void mousePressed(MouseEvent e) { 
report("mousePressed", e.paramString()); 
} 


public void mouseReleased(MouseEvent e) { 
report("mouseReleased”, e.paramString()); 
} 
}; 


MouseMotionListener mml = new MouseMotionListener() { 


public void mouseDragged(MouseEvent e) { 
report ("nouseDragged", e.paramString()); 
} 
public void mouseMoved(MouseEvent e) { 
report("mouseMoved". e.paramString()): 
} 
}; 
public MyButton(Color color, String label) { 
super (label); 
setBackground(color): 
addFocusListener(f1) ; 
addkeyListener (kl) ; 
addMouseListener (ml); 
addHouseMotionListener (mml) ; 





} 
} 


public TrackEvent() { 


‘mousePressed”, 
"mouseReleased", "mouseDragged", “mouseHoved" 
a 
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setLayout (new GridLayout(event.length + 1, 2)); 
for (String evt : event) { 
JTextField t = new JTextField(); 
t.setEditable(false) ; 
add(new JLabel(evt, JLabel.RIGHT)); 
add(t); 
h.put(evt, t); 


} 
add(b1); 
add (b2) ; 


Public static void main(String{] args) { 
run(new TrackEvent(), 788, 500); 


} 
} Ii~ 
在 MyButton 的 构造 器 中 ， 调 用 SetBackground0 方 法 设置 按钮 的 颜色 。 所 有 的 监听 器 都 是 通 
过 简单 的 方法 调用 进行 注册 的 。 
TrackEvent 类 包含 一 个 HashMap， 它 用 来 存放 表示 事件 类 型 的 字符 串 ， 以 及 一 些 
JTextField， 每 个 JTextField 用 来 显示 和 相应 事件 有 关 的 信息 。 当 然 ， 这 种 对 应 关系 可 以 静态 生 
成 而 不 用 放 进 HashMap， 不 过 我 认为 你 会 同意 这 样 做， 因为 如 此 一 来 使 用 和 修改 会 容易 得 多 。 
尤其 是 ， 如 果 要 在 TrackEvent 中 加 入 或 删除 新 的 事件 类 型 ， 那 么 只 要 在 event 数 组 中 加 入 或 删除 
字符 串 即 可 ， 其 他 工作 将 自动 完成 。 
调用 report0 的 时 候 ， 将 传 给 它 事件 的 名 称 以 及 从 事件 中 得 到 的 参数 字符 串 。 它 使 用 外 部 类 
中 的 HaspMap 对 象 b 来 查找 与 事件 名 称 相关 联 的 JTextField， 然 后 把 第 二 个 参数 放 进 该 文本 域 。 ” [1331] 
运行 这 个 例子 很 有 趣 ， 由 此 可 以 观察 到 程序 中 事件 发 生 时 的 实际 情况 。 
练习 10: (6) 使 用 SwingConsole 编 写 一 个 applet 应 用 程序 , 添加 一 个 JButton 和 一 个 JTextField。 
编写 恰当 的 监听 器 : 如 果 按钮 获得 了 焦点 ， 键 和 的 字符 将 出 现在 JTextField 里 。 
练习 11，(4) 从 JButton 继 承 编写 一 个 新 的 按钮 。 每 当 按钮 按 下 的 时 候 ， 将 为 按钮 随机 选择 
一 种 颜色 。 随 机 生成 颜色 的 方法 ， 请 参考 (本 章 稍 后 的 ) ColorBoxesjava。 
练习 12，(4) 通过 加 入 处 理 新 事件 的 代码 ， 在 TrackEventjava 中 监听 新 的 事件 。 需 要 自己 决 
定 监听 的 事件 类 型 。 


22.8 Swing 组 件 一 览 


既然 已 经 理解 了 布局 管理 器 和 事件 模型 ， 那 么 现在 可 以 学 习 如 何 使 用 Swing 组 件 了 。 本 节 将 
引导 读者 大 致 浏览 一 下 Swing 组 件 ， 并 介绍 其 最 常 使 用 的 功能 。 每 个 例子 都 尽 可 能 小 ， 这 样 就 很 
容易 抽出 所 需 代 码 ， 将 其 应 用 到 自己 的 程序 中 。 

请 记 住 : 

D 通过 编译 和 运行 本 章 可 下 载 的 源 代码 (从 www.MindView.com 下 载 )， 可 以 很 容易 地 观察 
每 个 例子 在 执行 过 程 中 的 状态 。 

2) 来 自 java.sun.com 的 JDK 文 档 内 包含 了 Swing 所 有 的 类 和 方法 (这 里 只 演示 了 其 中 的 一 部 分 )。 

3) Swing 中 的 事件 使 用 了 很 好 的 命名 习惯 ， 所 以 对 于 某 种 类 型 的 事件 ， 很 容易 猜测 出 如 何 编 
写 和 安装 事件 的 处 理 程序 。 可 以 使 用 本 章 前 面 的 查找 程序 ShowAddListeners.java， 来 帮助 查询 
特定 的 组 件 。 1337 

4) 当 程序 变 得 复杂 的 时 候 ， 应 该 过 渡 到 使 用 GUI 构造 工具 。 

22.8.1 按钮 
Swing 提供 了 许多 类 型 的 按钮 。 所 有 的 按钮 ， 包 括 复 选 框 、 单 选 按钮 ， 甚 至 菜单 项 ， 都 是 从 




















[到 
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AbstractButton (因为 包含 了 菜单 项 ， 所 以 将 其 命名 为 “AbstractSelector” 或 者 其 他 概括 性 的 名 
字 似 乎 更 恰当 一 些 ) 继承 而 来 。 很 快 你 就 会 看 到 菜单 项 的 使 用 ， 下 面 的 例子 演示 了 几 种 按钮 : 


//: gui/Buttons.java 

// Various Swing buttons. 

import javax.swing.*; 

import javax.swing.border.*; 

import javax.swing.plaf.basic.*; 

import java.ant.*: 

import static net.mindview.util.SwingConsole.*; 


public class Buttons extends JFrame { 

private JButton jb = new JButton("JButton”) ; 

private BasicArrowButton 
up = new BasicArrowButton(BasicArrowButton.NORTH) , 
down = new BasicArrowButton(BasicArrowButton. SOUTH), 
right = new BasicArrowButton(BasicArrowButton.EAST) ， 
left = new BasicArrowButton(BasicArrowButton.WEST) ; 

public Buttons() { 、 
setLayout (new FlowLayout()); 
add(jb); 
add(new JToggleButton("JToggleButton")); 
add(new JCheckBox("JCheckBox")) ; 
add(new JRadioButton("JRadioButton")) : 
JPanel jp = new JPanel(); 
jp.setBorder (new TitledBorder ("Directions"); 

， jp.add (up) ; 

jp. add (down) ; 
jp.add (left); 
jp.add (right); 
add(jp): 


public static void main(String[] args) { 
run(new Buttons(), 350, 208); 
) 

} H~ 

程序 开始 加 入 了 来 自 javax.swing.plaf.basic 的 BasicArrowButton， 然 后 又 加 入 了 几 种 不 同类 
型 的 按钮 。 运 行 例 子 ， 你 会 发 现 触发 器 按钮 (JToggleButton) 能 保持 自身 最 新 的 状态 ， 按 下 或 
者 弹出 。 不 过 复 选 框 和 单 选 按钮 看 起 来 差不多 ， 也 都 是 在 开 和 关 之 间 切换 (它们 都 是 从 
JrToggleButton 继 承 而 来 ) 。 

按钮 组 

要 想 让 单 选 按钮 表现 出 某 种 “ 排 它 ”行为 ， 必 须 把 它们 加 入 到 一 个 “按钮 组 ”(ButtonGroup) 
中 。 不 过 ， 正 如 下 面 的 例子 所 演示 的 ， 任 何 AbstractButton 对 象 都 可 以 加 入 到 按钮 组 中 。 

为 了 避免 重复 编写 大 量 的 代码 ， 下 面 这 个 例子 使 用 了 反射 功能 来 产生 几 组 不 同类 型 的 按钮 。 
注意 makeBPanel0 方 法 ， 它 用 来 创建 一 个 按钮 组 和 一 个 JPanel， 此 方法 的 第 二 个 参数 是 一 个 字 
符 串 数组 。 针 对 其 中 每 个 字符 串 ， 将 创建 一 个 由 第 一 参数 所 代表 的 按钮 实例 ， 然 后 将 此 按钮 加 
人 到 JPanel 中 : 

//: gui/ButtonGroups. java 

// Uses reflection to create groups 

// of different types of AbstractButton. 

‘import javax.swing.*; 

import javax.swing.border.*; 

import java.awt.*; 


import java. lang.reflect.*; 
import static net.mindview.util.SwingConsole.*; 


public class ButtonGroups extends JFrame { 
private static String{] ids = { 
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"June", "Ward", "Beaver", “Wally”, "Eddie", “Lumpy” 


}; 
static JPanel makeBPanel( 
Class<? extends AbstractButton> kind, String[] ids) { 
ButtonGroup bg = new ButtonGroup(); 
JPanel jp = new JPanel (); 
String title = kind.getName() : 
title = title.substring(title.lastIndexOf('.") + 1); 
jp.setBorder (new TitledBorder(title)); 
for(String id : ids) { 
AbstractButton ab = new JButton("failed”): 





try { 
7/ Get the dynamic constructor method B4 
J1 that takes a String argument: 
Constructor ctor = 
kind. getConstructor (String.class): 
// Create a new object: 
ab = (AbstractButton)ctor .newInstance(id) ; 
catch(Exception ex) { 
System.err.printIn("can't create " + kind); 


bg. add(ab) ; 
jp.add(ab) ; 


return jp; 


} 

public ButtonGroups() { 
setLayout(new FlowLayout()); 
add (makeBPanel (JButton.class, ids)); 
add (makeBPanel (JToggleButton.class, ids)); 
add (makeBPanel (JCheckBox.class, ids); 
add(makeBPanel (JRadioButton.class, ids)); 


} 
public static void main(String[] args) 《 
run(new ButtonGroups(), 500, 350); 


) 
y H~ 
边框 的 标题 是 从 类 的 名 称 中 得 到 的 ， 并 且 去 掉 了 其 中 的 路 径 信息 。AbstractButton 被 初始 化 
为 一 个 标签 为 “Failed” 的 JButton 对 象 ， 所 以 即使 你 忽略 了 异常 ,仍旧 能 够 在 屏幕 上 观察 到 失败 。 
getConstructor0 方 法 产生 一 个 Constructor 对 象 ， 这 个 构造 器 对 象 接受 “传递 给 getConstructor0 
的 Class 列 表 里 面 指定 的 类 型 ”所 组 成 的 数组 作为 参数 。 然 后 你 要 做 的 就 是 调用 newInstance0， 
并 且 把 包含 实际 参数 列表 传递 给 它 ， 在 本 例 中 就 是 ids 数 组 中 的 字符 串 。 
要 想 通过 按钮 得 到 “ 排 它 ”行为 ， 就 得 先 创建 一 个 按钮 组 ， 然 后 把 你 希望 具有 “ 排 它 ” 行 
为 的 按钮 加 入 到 这 个 按钮 组 中 。 运 行程 序 ， 你 将 发 现 除了 JButton 以 外 ， 其 他 按钮 都 具有 了 这 种 
“ 排 它 ” 行 为 。 
22.8.2 图 标 
可 以 在 JLable 或 者 任何 从 AbstractButton (包括 JButton、JCheckBox、JRadioButton 以 及 [1335] 
几 种 不 同 JMenultem) 继承 的 组 件 中 使 用 Icon。 和 JILabel 一 起 使 用 Icon 的 做 法 非常 直接 (后面 有 
例子 )。 下 面 的 例子 还 研究 了 与 按钮 (或 者 从 按钮 继承 的 组 件 ) 搭配 使 用 图 标的 所 有 方式 。 
可 以 使 用 任何 想 用 的 GIF 文件 ， 本 例 中 使 用 的 文件 来 自 于 本 书 的 源 代 码 包 (可 以 从 
| www.MindView.com 下 载 )。 要 打开 一 个 文件 并 且 得 到 图 形 ， 只 需 创建 一 个 Imagelcon 对 象 并 把 文 
| 件 名 传递 给 它 即 可 。 然 后 ， 就 能 在 程序 中 使 用 得 到 的 图 标 了 。 
//: gui/Faces.java 
// Icon behavior in JButtons. 


import javax.swing.*; 
import java.awt.*; 
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import java.awt.event.*; 
import static net.mindview.util.SwingConsole.*; 


public class Faces extends JFrame { 

private static Icon[] faces; 

private JButton jb, jb2 = new JButton("Disable”); 

private boolean mad = false; 

public Faces() { 

faces = new Icon[]{ 

new ImageIcon(getClass() .getResource("Face®.gif")), 
new ImageIcon(getClass() .getResource("Facel.gif")), 
new ImageIcon(getClass() .getResource("Face2.gif")), 
new ImageIcon(getClass() .getResource("Face3.gif")), 
new ImageIcon(getClass().getResource("Face4.gif")). 


}; 
jb = new JButton("JButton", faces[3]); 
setLayout (new FlowLayout()); 
jb.addActionListener(new ActionListener() { 
public void actionPerformed(ActionEvent e) { 
if(mad) { 
jb.setIcon(faces(3]); 
mad = false; 
} else { 
jb.setIcon(faces[0]); 
mad = true; 
} 
jb.setVerticalAlignment (JButton. TOP) ; 
jb. setHorizontalAlignment (JButton. LEFT) ; 
} 
1336 D: 
jb.setRolloverEnabled(true); 
jb.setRolloverIcon(faces[1]) 
jb. setPressedIcon(faces[2}); 
jb. setDisabledIcon(faces(4]); 
.SetToolTipText ("Yow!"); 
add(jb); 
jb2.addActionListener(new ActionListener () { 
public void actionPerformed(ActionEvent e) { 
1f(jb.isEnabled()) { 
jb.setEnabled(false) ; 
jb2.setText ("Enable"); 
) else { 
jb.setEnabled(true) ; 
jb2.setText("Disable"); 

















S 


} 
} 


D: 
add(jb2); 


public static void main(String[] args) { 
run(new Faces(), 250, 125); 


} 
} Mi~ ' 


许多 不 同 的 Swing 组 件 的 构造 器 都 接受 Icon 类 型 的 参数 ， 也 可 以 使 用 setIcon0 来 加 入 或 者 改 
变 图 标 。 本 例 还 演示 了 如 何 让 JButton (或 者 任何 AbstractButton 类 型 ) 在 各 种 情况 下 显示 不 同 
的 图 标 : 按 下 、 禁 止 ， 或 者 “浮动 ”( 鼠 标 移动 到 按钮 上 没有 点 击 的 时 修 )。 这 使 得 按钮 具有 了 


相当 不 错 的 动画 效果 。 
22.8.3 工具 提示 


前 面 的 例子 给 按钮 添加 了 一 个 “工具 提示 ”。 用 来 创建 用 户 接口 的 类 ， 绝 大 多 数 都 是 从 
JComponet 派 生 而 来 的 ， 它 们 包含 了 一 个 setToolTipText(String) 方法 。 所 以 ， 对 于 要 放置 在 窗 


| 体 上 的 组 件 ， 基 本 上 所 要 做 的 就 是 (对 于 任何 JComponet 派 生 类 的 对 象 jie) 像 这 样 编写 : 
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jc.setToolTipText ("Hy tip"); 
当 鼠 标 停留 在 这 个 JComponent 上 经 过 一 段 预先 指定 的 时 间 之 后 ， 在 鼠标 旁边 弹出 的 小 方 框 
里 就 会 出 现 你 所 设 定 的 文字 。 1337) 


22.8.4 文本 域 
下 面 的 例子 演示 了 JTextField 组 件 具 有 的 其 他 功能 : 


/1: gui/TextFields.java 
1/ Text fields and Java events. 
import javax.swing.*; 

import javax.swing.event 
import javax.swing. text. 
import java.awt.*; 

import java.awt.event 
import static net.mindview.util.SwingConsole.*; 




















public class TextFields extends JFrame { 

private JButton 
bl = new JButton("Get Text"), 
b2 = new JButton("Set Text"); 

private JTextField 
tl = new JTextField(30). 
t2 = new JTextField(30), 
t3 = new JTextField(30); 

private String s = ""; 

private UpperCaseDocument ucd = new UpperCaseDocument(): 

public TextFields() { 
t1.setDocument (ucd) ; 
ucd. addDocumentListener(new T1()): 
bl.addActionListener(new B1()); 
b2.addActionListener (new B2()) 
tl.addActionListener(new T1A()): 
setLayout(new FlowLayout()) ; 
add(b1) ; 
add (b2) ; 
add (tl); 
add(t2); 
add(t3); 





} 
class T1 implements DocumentListener { 
public void changedUpdate(DocumentEvent e) {} 
public void insertUpdate(DocumentEvent e) { 
t2.setText (t1.getText()): 
t3.setText("Text: "+ tl.getText()); 
} 
public void removeUpdate(DocumentEvent e) { 
t2.setText(tl.getText()); 1338) 














} 


Class TIA implements ActionListener { 
private int count = @; 
public void actionPerformed(ActionEvent e) { 
t3.setText("t1 Action Event ”+ count++); 
} 
} 
class B1 implements ActionListener { 
public void actionPerformed(Action€vent e) { 
if (t1.getSelectedText() == null) 
s = tl.getText(); 
else 
5 = tl.getSelectedText(); 
tl. setEditable(true); 
} 





} 
class B2 implements ActionListener { 











1339} 
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public void actionPerformed (ActionEvent e) { 
ucd. setUpperCase(false) ; 
t1.setText("Inserted by Button 2: " + s); 
ucd. setUpperCase(true) ; 
t1.setEditable(false); 

} 


} 
public static void main(String[] args) { 
run(new TextFields(), 375, 268); 
} 
} 


class UpperCaseDocument extends PlainDocument { 
private boolean upperCase = true; 
public void setUpperCase(boolean flag) { 
upperCase = flag; 
} 
public void 
insertString(int offset. String str, AttributeSet attSet) 
throws BadLocationException { 
if(upperCase) str = str.toUpperCase(); 
super. insertString(offset, str, attSet); 


} 
} Mi~ 


当 JTextField 对 象 t 的 动作 监听 器 被 触发 时 ,JTextField 对 象 3 是 要 被 告知 该 事件 的 对 象 之 一 。 
可 以 观察 到 ， 只 有 当 按 下 “ 回 车 ” 键 的 时 候 ，JTextField 的 动作 监听 器 才 会 被 触发 。 

JTextField 对 象 tL 关 联 了 多 个 监听 器 。T1 是 一 个 DocumentListener， 用 来 对 “文档 ”( 本 例 
中 指 JTextField 的 内 容 ) 中 的 变化 作出 反应 。 它 将 自动 把 t 的 文本 复制 到 t2。 此 外 ，tl 的 文档 被 
设置 成 PlainDocument 的 派生 类 对 象 ， 就 是 代码 中 的 UpperCaseDocument， 它 把 所 有 字符 强制 
变 成 大 写 。 此 外 ， 它 还 能 自动 检测 退 格 键 ， 并 执行 删除 、 调 整 插 字符 以 及 处 理 你 所 期 望 的 所 有 
行为 。 

练习 13，(3) 修改 TextFieldsjava， 使 得 t2 里 面 的 字符 保持 原来 输入 时 候 的 大 小 写 ， 而 不 要 
自动 转换 成 大 写 。 
22.8.5 边框 

JComponent 有 一 个 setBorder0 方 法 ， 它 允许 你 为 任何 可 视 组 件 设置 各 种 边框 。 下 面 的 例子 
使 用 showBorder() 方 法 演示 了 一 些 可 用 的 边框 。 此 方法 先 创建 一 个 JPanel， 然 后 设置 相应 的 边 
框 。 此 外 ， 它 还 使 用 RTTI (运行 时 类 型 识别 ) 来 得 到 正在 使 用 的 边框 名 称 (去 掉 了 路 径 信息 )， 
然后 把 这 个 名 称 放 进 面板 中 间 的 一 个 JLabel 中 : 


//: gut/Borders. java 

// Different Swing borders. 

import javax.swing.*; 

import javax.swing.border.*; 

import java.awt.*; 

import static net.mindview.util.SwingConsole.*; 


public class Borders extends JFrame { 
static JPanel showBorder (Border b) { 
JPanel jp = new JPanel(); 
jp.setLayout (new BorderLayout()); 
String nm = b.getClass().toString(); 
nm = nm.substring(nm.lastIndexOf(’.') + 1); 
jp.add(new JLabel(nm, JLabel.CENTER), 
BorderLayout .CENTER) ; 
jp.setBorder (b) ; 
return jp; 





Doa w Ay a d 
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public Borders() { 

setLayout(new GridLayout(2,4)); 
add(showBorder (new TitledBorder("Title"))); 
add (showBorder (new EtchedBorder())); 
add (showBorder (new LineBorder (Color.BLUE))); 
add (showBorder ( 

new MatteBorder(5,5,30,30,Color.GREEN))); 
add (showBorder ( 

new BevelBorder (BevelBorder .RAISED))) ; 
add (showBorder ( 

new SoftBevelBorder (BevelBorder . LOWERED) )) ; 
add (showBorder (new CompoundBorder ( 

new EtchedBorder(), 

new LineBorder (Color .RED)))); 


} 
Public static void main(String[] args) { 
run(new Borders(), 580, 300); 


} 
} Mi~ 


也 可 以 自己 编写 边框 代码 ， 然 后 把 它们 加 入 到 按钮 、 标 签 等 任何 从 JComponent 派 生 的 组 件 
中 去 。 
22.8.6 一 个 迷你 编辑 器 

JTextPane 控 件 可 以 毫 不 费事 地 支持 许多 编辑 操作 。 下 面 的 例子 是 对 这 个 组 件 的 简单 应 用 ， 
其 中 忽略 了 该 组 件 所 能 提供 的 其 他 的 大 量 功能 ; 


/1:_gui/TextPane. java 
// The JTextPane control is a little editor. 
import javax.swing.*; 
import java.awt.*; 
import java.awt.event.*; 
import net.mindview.util.*; 
import static net.mindview.util. SwingConsole.*; 








public class TextPane extends JFrame { 
private JButton b = new JButton("Add Text"); 
private JTextPane tp = new JTextPane(); 
private static Generator sg = 
new RandomGenerator .String(7) ; 





public TextPane() { 
b.addActionListener(new Actiontistener() { 





1341| 








public void actionPerformed(ActionEvent e) { 
for(int 1 = 1; 1 < 10; i++) 
tp.setText(tp.getText() + sg.next() + "\n"); 
} 


add(new JScrollPane(tp)) ; 
add (BorderLayout.SOUTH, b); 


} 
public static void main(String{] args) { 
run(new TextPane(), 475, 425); 


} line 


| 按钮 的 功能 只 是 添加 一 些 随机 生成 的 文本 。JTextPane 的 目的 是 提供 即时 编辑 文本 的 功能 ， 

所 以 这 里 没有 append0 方 法 。 在 本 例 中 〈 坦 白地 说 ， 这 不 是 一 个 可 以 发 挥 JTextPane 功 能 的 好 例 

子 ), 文 本 必须 被 捕获 并 修改 ， 然 后 使 用 setText0 将 其 放 回 到 文本 面板 中 。 

| 各 个 元 素 是 通过 使 用 JErame 默 认 的 BorderLayouti 添 加 到 JFrame 中 的 ， 而 JIextPane 被 泊 

加 (到 JSerollpane 中 ) 时 ， 没 有 指定 其 区 域 ， 因 此 它 将 从 中 间 开始 填充 面板 ， 直 到 与 边框 对 齐 。 

| JButton 被 添加 到 了 SOUTH， 因 此 所 有 组 件 将 被 调整 到 这 个 区 域内 ， 本 例 中 ， 按 钮 将 处 于 屏 划 
的 底部 。 


| 
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注意 ，JTextPane 还 有 诸如 自动 换行 这 样 的 内 置 功能 。 其 他 的 功能 可 以 参考 JDK 文 档 。 

练习 14: (2) 修改 TextPanejava， 要 求 使 用 JTextArea 而 不 是 JTextPane。 
22.8.7 复 选 框 

复 选 框 提供 了 一 种 做 出 “选中 ”或 “不 选 ”单一 选择 的 方式 。 它 包含 了 一 个 小 方 框 和 一 个 
标签 。 这 个 方 框 中 通常 是 有 一 个 “x” 标 记 (或 者 其 他 能 表明 “选中 ”的 标记 ) 或 者 为 空 ， 这 取 
决 于 复 选 框 是 否 被 选中 。 

通常 会 使 用 接受 标签 作为 参数 的 构造 器 来 创建 JCheckBox。 可 以 获取 和 设置 状态 ， 也 可 以 
获取 和 设置 其 标签 ， 甚 至 可 以 在 JCheckBox 对 象 已 经 建立 之 后 改变 标签 。 

当 JCheckBox 被 选中 或 清理 选中 时 ， 将 发 生 一 个 事件 ， 你 可 以 用 与 对 付 按钮 相同 的 方式 来 

1342] 捕获 这 个 事件 : 使 用 ActionListener。 在 下 面 的 例子 中 ， 将 枚 举 所 有 被 选中 的 复 选 框 ， 然 后 在 

JTextArea 里 显示 : 


//: Bui/CheckBoxes .java 
// Using JCheckBoxes. 
import javax.swin 
import java.awt.*; 

import java.awt.event.*; 

import static net.mindview.util.SwingConsole.*; 




















public class CheckBoxes extends JFrame { 
private JTextArea t = new JTextArea(6, 15); 
private JCheckBox 
cbl = new JCheckBox("Check Box 1"), 
cb2 = new JCheckBox("Check Box 2 
cb3 = new JCheckBox("Check Box 3"); 
public CheckBoxes() ( 
cb1.addActionListener(new ActionListener() { 
public void actionPerformed(ActionEvent e) { 
trace("1", cb1); 
} 
H: 
cb2.addActionListener (new ActionListener() { 
public void actionPerformed(ActionEvent e) { 
trace("2", cb2); 
) 





H: 
cb3.addActtonListener (new ActionListener () { 
public void actionPerformed(ActionEvent e) { 
trace("3", cb3); 
} 
Ye 
setLayout(new FlowLayout()): 
add(new JScrollPane(t)); 
add(cb1); 
add (cb2); 
add(cb3); 
} 
private void trace(String b, JCheckBox cb) { 
if (cb. isSelected()) 
t.append("Box ”+ b + " Set\n"); 
else 
t.append("Box " + b + " Cleared\n"); 
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public static void main(String(} args) { 
run(new CheckBoxes(), 266, 308): 


} 
} H~ 


| trace( 方 法 使 用 append0 把 所 选 的 JCheckBox 的 名 称 及 其 当前 状态 显示 到 JTextArea 中 ， 所 
以 可 以 看 到 一 个 复 选 框 列表 ， 其 中 包括 了 复 选 框 名 称 及 其 状态 。 
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练习 15: (5) 向 练习 5 编写 的 应 用 程序 中 添加 一 个 复 选 框 ， 捕 获 其 事件 ， 并 在 事件 处 理 程序 
中 向 文本 域 插入 不 同 的 文字 。 
22.8.8 单 选 按钮 

GUI 编程 中 单 选 按钮 的 概念 来 源 于 电子 按钮 发 明之 前 汽车 上 收音 机 使 用 的 机 械 按钮 ， 当 按 下 
其 中 的 一 个 ， 其 他 被 按 下 的 技 钮 将 被 弹出 。 所 以 ， 单 选 按钮 强制 你 在 多 个 选项 中 只 能 选择 一 个 。 

要 设置 一 组 关联 的 JRadioButton， 你 需要 把 它们 加 入 到 一 个 ButtonGroup 中 〈 窗 体 上 可 以 有 
任意 数目 的 ButtonGroup) 。 可 以 选择 将 其 中 的 一 个 按钮 设置 为 选中 (true) (在 构造 器 的 第 二 个 
参数 中 设置 )。 如 果 你 把 多 个 单 选 按钮 的 状态 都 设置 为 选中 ， 那 和 只 有 最 后 设置 的 那个 有 效 。 

下 面 是 使 用 了 单 选 按钮 的 简单 例子 它 展示 了 使 用 ActionListener 来 捕获 事件 ; 


//: gui/RadioButtons. java 

// Using JRadioButtons 

import javax.swing.*; 

import java.awt.*; 

import java.awt.event.*; 

‘import static net.mindview.util.SwingConsole.*; 


public class RadioButtons extends JFrame { 
private JTextField t = new JTextField(15): 
private ButtonGroup g = new ButtonGroup(); 
Private JRadioButton 
rbl 4 new JRadioButton("one", false), 
rb2 = new JRadioButton("two", false), 
rb3 = new JRadioButton("three", false); 
private ActionListener al = new ActionListener() { 
public void actionPerformed(ActionEvent e) { 
t.setText("Radio button " + 


((JRadioButton)e.getSource()).getText()); 


k 

Public RadioButtons() { 
rb1. addAct ionListener (al); 
rb2. addAct fonListener (al); 
rb3.addAct ionListener (al); 
g.add(rb1); g.add(rb2); g.add(rb3); 
t.setEditable(false) ; 
setLayout (new FlowLayout()); 
add(t); 
add (rb1); 
add (rb2); 
add(rb3); 

} 

public static void main(String[] args) { 
run(new RadioButtons(), 200, 125); 


} 

} i~ 

这 里 使 用 了 文本 域 来 显示 状态 。 因 为 它 仅仅 用 来 显示 而 不 是 收集 数据 ， 所 以 被 设置 成 “不 
可 编辑 "。 因 此 ， 这 是 可 以 用 来 代替 JLabel 的 一 种 方式 。 
22.8.9 组 合 框 

与 一 组 单 选 按钮 的 功能 类 似 ， 组 合 框 (下拉 列表 ) 也 是 强制 用 户 从 一 组 可 能 的 元 素 中 只 选 
择 一 个 。 不 过 ， 这 种 方法 更 加 紧凑 ， 而 且 在 不 会 使 用 户 感到 迷惑 的 前 提 下 ， 改 变 下 拉 列 表 中 的 
内 容 更 容易 〈 当 然 也 可 以 动态 改变 单 选 按钮 ， 不 过 这 么 做 显然 易 造成 冲突 )。 

默认 状态 下 ，JComboBox 组 合 框 与 Windows 操 作 系 统 下 的 组 合 框 并 不 完全 相同 ， 后 者 允许 
从 列表 中 选择 ， 或 者 自己 输入 。 要 想得到 这 样 的 行为 ， 必 须 调用 setEditable() 方 法 。 使 用 
JComboBox 组 合 框 ， 你 能 且 只 能 从 列表 中 选择 一 个 元 素 。 在 下 面 的 例子 里 ，JComboBox 组 合 框 
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//: gui/ComboBoxes . java 
71 Using drop-down lists. 

import javax.swing.*; 

import java.awt.*; 

import java.awt.event.*; 

import static net.mindview.util.SwingConsole.*; 


public class ComboBoxes extends JFrame { 
private Stringf] description = { 


"Ebullient", "Obtuse", "Recalcitrant", "Brilliant", 


"Somnescent", “Timorous”, "Florid", “Putrescent” 
}; 
private JTextField t = new JTextField(15); 
private JComboBox c = new JComboBox(); 
private JButton b = new JButton("Add items"); 
private int count = @; 
public ComboBoxes() { 
for(int 1 = 0; 1 < 4; i++) 

c.addI tem(description{count++)) ; 
t.setEditable(false) ; 
b.addActionListener(new ActionListener() { 

public void actionPerformed(ActionEvent e) { 

if(count < description. length) 
c.addI tem(description{count++] ) ; 
} 


H: 
c.addActionListener (new ActionListener() { 
public void actionPerformed (ActionEvent e) { 
t.setText("index: "+ c.getSelectedIndex() + * 


((JComboBox)e.getSource()).getSelectedItem()); 


} 


H: 

setLayout(new FlowLayout()); 
add(t); 

add(c); 

add (b) ; 


} 
public static void main(String[] args) { 
run(new ComboBoxes(), 268, 175); 


} 
} Mi~ 


上 例 中 的 JTextField 被 用 来 显示 “被 选中 的 索引 ”( 当 前 被 选中 元 素 的 序号 )， 以 及 组 合 框 中 


被 选中 元 素 的 文本 。 


时 ， 会 出 现下 拉 列 表 ， 而 JList 总 是 在 屏幕 上 占 
到 列表 框 中 被 选中 的 项 目 ， 只 需 调 用 getSelectedValues0， 它 可 以 产生 一 个 字符 串 数组 ， 里 面 是 
被 选中 的 项 目 名 称 。 
JIList 组 件 允 许多 重 选择 ， 要 是 按 住 Ctrl 键 ， 连 续 在 多 个 项 目 上 单 击 ， 那 么 原先 被 选中 的 项 目 
仍旧 保持 选中 状态 ， 也 就 是 说 可 以 选中 任意 多 的 项 目 。 如 果 选 中 了 某 个 项 目 ， 按 住 “Shift” 键 
并 单 击 另 一 个 项 目 ， 那 么 这 两 个 项 目 之 间 的 所 有 项 目 都 将 被 选中 。 要 从 选中 的 项 目 组 中 去 掉 一 
， 可 以 按 住 Ctrl 键 在 此 项 目 上 单 击 。 


22.8.10 列表 框 
列表 框 和 JComboBox 组 合 框 明显 不 同 ， 这 不 仅仅 体现 在 外 观 上 。 当 激活 JComboBox 组 合 框 
定 行 数 的 空间 ， 大 小 也 不 会 改变 。 如 果 要 得 





11: gui/List.java 
import javax.swing.*; 

import javax.swing.border.*; 
import javax.swing.event.*; 


开始 时 已 经 有 了 一 些 元 素 ， 然 后 当 一 个 按钮 按 下 的 时 候 ， 将 向 组 合 框 中 加 入 新 的 元 素 。 
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import java.ant.*: 
import java.awt.event.*: 
import static net.mindview.util.SwingConsole.*; 


public class List extends JFrame { 
private String[] flavors = { 
“Chocolate”, "Strawberry", "Vanilla Fudge Swirl”, 
"Mint Chip", "Mocha Almond Fudge, "Rum Raisin”, 
“Praline Cream", "Mud Pie” 
Fi 
private DefaultListModel lItems = new DefaultListModel(); 
private JList Ist = new JList(lItems); 
private JTextArea t = 
new JTextArea(flavors.length, 20); 
private JButton b = new JButton("Add Item"); 
private ActionListener bl = new ActionListener () { 
public void actionPerformed(ActionEvent e) { 
if(count < flavors. length) { 
Lltems.add(@, flavors[count++]); 
} else { 
// Disable, since there are no more 
// flavors left to be added to the List 
b.setEnabled (false); 
} 
} 


private ListSelectionListener 11 = 
new ListSelectionListener() { 
public void valueChanged(ListSelectionEvent e) { 
if(e.getValueIsAdjusting()) return; 
t.setText(""); 
for (Object item : 1st.getSelectedValues()) 
t.append(item + "\n"); 
} 


}; 
private int count = 0; 
public List) { 
t.setEditable(false); 
setLayout (new FlowLayout()); 
// Create Borders for components: 
Border brd = BorderFactory.createNatteBorder( 
1, 1, 2, 2, Color. BLACK) ; 
Ist. setBorder (brd); 
t. setBorder (brd) ; 
// Add the first four items to the List 
for(int 1 = 0; i < 4; i++) 
LItems addélement (flavors (count++]) ; 
add(t) ; 
add(1st); 
add(b) ; 
// Register event listeners 
Ast addL istSelectionListener (11); 
b.addAct ionL istener (b1) ; 
} 
public static void main(String{] args) ( 
run(new List(), 250, 375); 
} 
dh 


可 以 观察 到 列表 框 周围 添 加 了 边框 。 

如 果 只 是 要 把 一 个 字符 串 数组 加 入 JList， 那 么 有 一 个 更 简单 的 办 法 ， 只 要 把 数组 传递 给 
JList 的 构造 器 ， 就 能 自动 构造 列表 框 。 上 例 中 使 用 “列表 模型 ”的 唯一 原因 是 ， 这 样 可 以 在 程 
序 执行 的 过 程 中 操纵 列表 框 。 

JList 本 身 没有 对 滚动 提供 直接 的 支持 。 当 然 ， 你 要 做 的 只 是 把 JList 包 装 进 JSerollPane， 它 
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将 自动 帮 你 处 理 其 中 的 细节 。 

练习 16: (5) 通过 传递 数组 给 构造 器 ， 以 及 移 除 动态 地 向 列表 框 添加 元 素 的 代码 ， 来 简化 
List.java, 
22.8.11 页 签 面板 

JTabbedPane 克 许 你 创建 “页 签 式 的 对 话 框 *， 这 种 对 话 框 中 沿 着 窗 体 的 一 边 有 类 似 文件 夹 
的 页 签 ， 当 你 在 页 签 上 点 击 时 ， 就 会 向 前 进入 到 另 一 个 不 同 的 对 话 框 中 。 


/1: gui/TabbedPanel.java 
// Demonstrates the Tabbed Pane. 

import javax.swing.*; 

import javax.swing.event.*; 

import java.awt.*; 

import static net.mindview.util.SwingConsole.*; 


public class TabbedPanel extends JFrame { 
Private String[] flavors = { 
"Chocolate", "Strawberry", "Vanilla Fudge Swirl", 
“Mint Chip", "Mocha Almond Fudge", "Rum Raisin", 
“Praline Cream", "Mud Pie” 
j 
private JTabbedPane tabs = new JTabbedPane(); 
private JTextField txt = new JTextField(20); 
Public TabbedPanel() { 
int 1 = 0; 
for (String flavor : flavors) 
tabs. addTab(flavors{i], 
new JButton("Tabbed pane ”+ i++)); 
tabs. addChangeListener (new Changelistener() { 
public void stateChanged(ChangeEvent e) { 
txt.setText("Tab selected: " + 
tabs .getSelectedIndex()); 





} 
Di 
add(BorderLayout. SOUTH, txt); 
add(tabs) ; 


Public static void main(Stringl] args) { 
run(new TabbedPanel(), 400, 250); 
} 
AAA 


在 运行 程序 的 时 候 可 以 观察 到 ， 如 果 页 签 太 多 ， 即 在 一 行 中 放 不 下 它们 的 时 候 ，JTabbed- 
Pane 能 够 自动 把 页 签到 起 来 。 如 果 是 在 命令 行 方式 下 运行 该 程序 ， 可 以 通过 调整 窗口 大 小 来 观察 。 
22.8.12 消息 框 

视窗 环境 下 通常 包含 了 一 组 标准 的 消息 框 ， 使 得 能 够 快速 地 把 消息 通知 给 用 户 ， 或 者 是 从 
用 户 那里 得 到 信息 。 在 Swing 中 ， 这 些 消息 框 包含 在 JOptionPane 组 件 里 。 你 有 许多 选择 (有 些 
非 党 高级) ， 但 最 常用 的 可 能 就 是 消息 对 话 框 和 确认 对 话 框 ， 它 们 分 别 可 以 通过 调用 静态 的 
JOptionPane.showMessageDialog0 和 JOptionPane.showConfirmDialog(0 方 法 得 到 。 下 面 的 例子 
演示 了 JOptionPane 中 一 些 可 用 的 消息 框 。 


//: gui /MessageBoxes. java 
// Demonstrates JOptionPane. 

import javax.swing.*: 

“import java.awt.*; 

import java.awt.event.*; 

import static net.mindview.util.SwingConsole.*; 


public class MessageBoxes extends JFrame { 


| a ae 
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private JButton{] b = { 
new JButton("Alert"), new JButton("Yes/No"), 
new JButton("Color"), ‘new JButton("Input"), 
new JButton("3 Vals") 





a 
private JTextField txt = new JTextField(15); 
private ActionListener al = new ActionListener() { 
public void actionPerformed(ActionEvent e) { 
String id = ((JButton)e.getSource()).getText(); 
if (id -equals("Alert")) 

JOptionPane. showMessageDialog(ni 
“There's a bug on youl", "He 
JOptionPane.ERROR_MESSAGE) ; 

else if(id.equals("Yes/No")) 

JOptionPane. showConfirmDialog(null, 
“or no", “choose yes", 350) 
JOptionPane.YES_NO_OPTION) ; 

else if(id.equals("Color")) { 
Object{] options = { "Red", “Green” }; 
int sel = JOptionPane. showOptionDialog( 
null, "Choose a Color!", "Warning", 
JOptionPane.DEFAULT_OPTION, 
JOptionPane.WARNING MESSAGE, null, 
options, options{@]); 
if (sel != JOptionPane.CLOSED_OPTION) 
txt.setText ("Color Selected: ”+ options[sel]); 
else if(id.equals("Input")) { 
String val = JOptionPane. showInputDi alog( 
"How many fingers do you see?"); 
txt.setText (val); 
else if(id.equais("3 Vals")) { 
Object[] selections = {"First", "Second", "Third"}: 
Object val = JOptionPane. showInputDialog( 
null, "Choose one, "Input", 
JOptionPane. INFORMATION MESSAGE, 
null, selections, selections[@]); 
if(vat 1= null) 
txt.setText (val. toString()); 





1, 








} 
} 


}; 
public MessageBoxes() { 
setLayout (new FlowLayout()): 
for(int 1 = 8; i < b.length; i++) { 
b[i] .addActionListener(al); 
add(b[i]); 


} 
add(txt); 


} 
public static void main(String[] args) { 
run(new MessageBoxes(). 208, 200); 


) 
dM hw 
为 了 只 编写 单一 的 ActionListener， 我 使 用 了 “检查 按钮 上 字符 串 标签 ”的 方法 来 判断 事件 
的 来 源 ， 这 有 点 冒险 。 其 问题 在 于 标签 可 能 会 有 拼写 错误 ， 尤 其 是 大 小 写 ， 这 种 缺陷 很 难 发 现 。 [5 
注意 ，showOptionDialog0 和 showInputDialog0 方 法 提供 了 返回 对 象 ， 此 对 象 包含 了 用 户 输 
入 的 信息 。 
练习 17: (5) 使 用 SwingConsole 编 写 一 个 应 用 程序 。 在 http://java.sun.com 的 JDK 文 档 中 ， 查 
找 JPasswordField 并 把 它 添加 到 程序 中 。 如 果 用 户 输入 了 正确 的 密码 ， 使 用 JOptionPane 向 用 户 
显示 “成 功 ” 的 消息 。 
练习 18: (4) 修改 MessageBoxes.java， 使 它 的 每 个 按钮 拥有 单独 的 ActionListener (而 不 是 
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通过 按钮 文字 的 匹配 来 共享 )。 
22.8.13 菜单 

每 个 能 够 持 有 菜单 的 组 件 ， 包 括 JApplet、JFrame、JDialog 以 及 它们 的 子 类 ， 它 们 都 有 一 个 
setJMenuBar( 方 法 ， 它 接受 一 个 JMenuBar 对 象 ( 某 个 特定 组 件 只 能 持 有 一 个 JMenuBar 对 象 ) 
作为 参数 。 你 先 把 JMenu 对 象 添加 到 JMenuBar 中 ， 然 后 把 JMenuItem 添 加 到 JMenu 中 。 每 个 
JMenuItem 都 能 有 一 个 相关 联 的 ActionListener， 用 来 捕获 菜单 项 被 选中 时 所 触发 的 事件 。 

在 Java 和 Swing 中 ， 必 须 在 源 代码 中 构造 所 有 的 菜单 。 下 面 是 个 非常 简单 的 菜单 例子 ， 


/1/: gui/SimpleMenus. java 

import javax.swing.*; 

import java.awt.*; 

import java.awt.event.*; 

import static net.mindview.util.SwingConsole.*; 


public class SimpleMenus extends JFrame { 
private JTextField t = new JTextField(15): 
private ActionListener al = new ActionListener() { 
public void actionPerformed(ActionEvent e) { 
t.setText(((JMenuItem)e.getSource()).getText()); 
} 


}; 

private JMenu[] menus = { £ 
new JHenu("Winken"), new JMenu("Blinken"), 
new JMenu("Nod”) 


Xi 

private JMenuItem[] items = { 
new JMenuItem("Fee"), new JMenuItem("Fi"), 
new JMenuItem("Fo"), new JMenultem(*Zip"), 
new JMenultem("Zap"), new JMenuItem("Zot"), 
new JMenuItem("Olly"), new JMenuItem("Oxen") , 
new JMenuItem("Free") 


i 
public SimpleMenus() { 
for(int i = 0; i < items.length; i++) { 
items[i}.addActionListener (al); 
menus [i % 3] .add(items[i]); 
} 
JMenuBar mb = new JMenuBar(); 
for(JMenu jm : menus) 
mb.add(jm); 
setJMenuBar (mb) ; 
setLayout(new FlowLayout()); 
add(t); 


} 
public static void mafn(String[] args) { 
run(new SimpleMenus(), 208, 156); 


} 

} H~ 

程序 中 通过 取 模 运算 “i%3” 把 菜单 项 分 配给 三 个 JMenu。 每 个 JMenultem 必 须 有 一 个 相 
关联 的 ActionListener， 这 里 使 用 了 同一 个 ActionListener， 不 过 通常 要 为 每 个 JMenuItem 单 独 
准备 一 个 ActionListener。 

JMenuItem 从 AbstractButton 继 承 而 来 ， 所 以 它 具 有 类 似 按钮 的 行为 。 它 提供 了 一 个 可 以 单 
独 放 置 在 下 拉 菜 单 上 的 条 目 。 还 有 三 种 类 型 继承 自 JMenuItem: JMenu 用 来 持 有 其 他 的 
JMenultem (这 样 才 能 实现 层 和 公式 菜单 ) ，JCheckBoxMenuItem 提 供 了 一 个 复 选 标记 ， 用 来 表 
明 菜单 项 是 否 被 选中 ，JRadioButtonMenultem 包 含 了 一 个 单 选 按钮 。 

下 面 是 一 个 更 复杂 的 创建 菜单 的 例子 ， 这 里 仍然 是 冰激凌 口味 的 例子 。 这 个 例子 还 演示 了 
层 释 式 菜单 、 键 盘 快 捷 键 、JCheckBoxMenulItem， 以 及 动态 改变 菜单 的 方法 : 





图形 化 用户 界面 799 





711: gui/Menus.java 
7/ Submenus，check box menu items, swapping menus, 

// mnemonics (shortcuts) and action commands. 

import javax.swing.*; 

import java.awt.*; 1353) 
import java.awt.event. 
import static net.mindview.util.SwingConsole.*; 











public class Menus extends JFrame { 
Private String[] flavors = { 
"Chocolate", “Strawberry”, "Vanilla Fudge Swirl", 
“Mint Chip", “Mocha Almond Fudge", “Rum Raisin", 
"Praline Cream", "Mud Pie" 
}; 
private JTextField t 
private JMenuBar mbl 
private JMenu 
f = new JMenu("File"), 
m = new JMenu("Flavors"), 
s = new JMenu("Safety"); 
// Alternative approach: 
private JCheckBoxMenuItem[] safety = ( 
new JCheckBoxMenul tem("Guard"), 
new JCheckBoxMenul tem("Hide") 





new JTextField("No flavor", 30); 
new JMenuBar(); 





a 
private JMenuItem{] file = { new JMenuItem(*Open") }; 
// A second menu bar to swap to: 
private JMenuBar mb2 = new JMenuBar() ; 
private JMenu fooBar = new JMenu(*fooBar"): 
private JMenuItem[] other = { 
// Adding a menu shortcut (mnemonic) is very 
// simple, but only JMenuItems can have them 
// in their constructors: 
new JMenultem("Foo", KeyEvent.VK_F), 
new JMenultem("Bar”, KeyEvent.VK_A), 
// No shortcut: 
new JMenultem("Baz"), 
Me 
private JButton b = new JButton("Swap Menus"); 
class BL implements ActionListener { 
public void actionPerformed(ActionEvent e) { 
JMenuBar m = getJMenuBar() ; 
setJMenuBar(m == mbl ? mb2 : mbl); 
validate(); // Refresh the frame 
} 





} 
class ML implements ActionListener { 
public void actionPerformed(ActionEvent e) { 
JMenuItem target = (JMenultem)e.getSource(); 1354) 
String actionCommand = target.getActionCommand() ; 
if (actionCommand.equals("Open")) { 
String s = t.getText(); 
boolean chosen = false; 
for(String flavor : flavors) 
if(s.equals(flavor)) 
chosen = true; 
if (ichosen) 
t.setText("Choose a flavor first! 
else 
t.setText("Opening " +s + ". Mmm, 
} 


} 











} 
class FL implements ActionListener { 
public void actionPerformed(ActionEvent e) { 
JMenuItem target = (JMenuItem)e.getSource(); 
t.setText (target. getText()); 
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} 


} 
// Alternatively, you can create a different 


71 class for each different Menultem. Then you 
// don't have to figure out which one it is: 
Class Fool implements ActionListener { 
public void actionPerformed(ActionEvent e) { 
t-setText("Foo selected"): 
} 
} 
Class Barl implements ActionListener { 
public void actionPerformed(ActionEvent e) { 
t.setText("Bar selected"); 
} 
} 
class BazL implements ActionListener { 
public void actionPerformed(ActionEvent e) { 
t.setText("Baz selected"); 
i } 
Class CHIL implements ItemListener { 
public void itemStateChanged(ItemEvent e) { 
ICheckBoxMenultem target = 
(JSCheckBoxHenul tem)e. getSource() ; 
String actionCommand = target.getActionCommand(); 
if (act ionCommand.equats(*Guard")) 
t.setText ("Guard the Ice Cream! " + 
“Guarding is ”+ target.getState()); 
else {f(actionCommand. equals("Hide")) 
t.setText("Hide the Ice Cream! ”+ 
“Is it hidden? ”+ target.getState()): 





} 


} 
public Menus() { 
ML ml = new MLO); 
CHIL cmil = new CHILO); 
safety [@] .setActionCommand ("Guard") ; 
safety [0] . setMnemonic (KeyEvent . VK_G: 
safety [0] .addItemListener (cmil 
safety[1] .setActionCommand("Hide" 
safety[1] .setHnemontc (KeyEvent.VK_H) ; 
safety[1] .addItemListener (cmil 
other [0] .addActionListener (new Fool ()); 
other [1] .addActionListener (new Barl ()) : 
other [2] .addActionListener (new BazL()) ; 
FL fl = new FLO; 
int n = 6; 
for(String flavor : flavors) { 
JMenuItem mi = new JMenultem(flavor) ; 
mi .addActionListener (f1); 
m.add(mi) ; 
// Add separators at intervals: 
if((nt+ + 1) % 3 == 0) 
m.addSeparator () 




















} 

for (JCheckBoxMenultem sfty : safety) 
s.add(sfty); 

S.setMnemonic (KeyEvent.VK_A); 

f.add(s); 

f.setMnemonic(Key€vent..VK_F) ; 

for(int i = @; i < file.length; i++) ( 
fileli] .addActionListener (ml); 
f.add(fileli]); 

} 

mbl.add(f); 

mb1.add (m 

setJMenuBar (mb1) ; 
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t.setEditable(false); 

add(t, BorderLayout.CENTER); 

1/ Set up the system for swapping menus: 

b.addActionListener(new BL()): 

b. setlnemonic(KeyEvent .VK_S); 

add(b, BorderLayout .NORTH) ; 

for(JMenuItem oth : other) 
fooBar.add(oth) ; 

fooBar . setMnemonic (KeyEvent .VK_B); 

mb2. add (fooBar) ; 


) 
public static void main(String{] args) { 
run(new Menus(), 300, 200); 


} 

} /LAI 

在 这 个 程序 中 ， 我 把 菜单 项 放 到 了 几 个 数组 中 ， 然 后 通过 遍历 这 些 数组 ， 并 为 每 个 
JMenultem 调 用 add0 方 法 的 方式 ， 将 它们 添加 到 菜单 中 。 这 种 方式 让 添加 或 减少 菜单 项 不 至 于 
KE. 

程序 中 不 是 创建 了 一 个 而 是 创建 了 两 个 JMenuBar， 用 以 演示 程序 运行 期 间 可 以 动态 替换 菜 
单条 。 你 可 以 看 到 如 何 用 JMenu 构 造 JMenuBar， 以 及 用 JMenuItem、JCheckBoxMenuItem 甚 
至 其 他 JMenu (产生 子 菜单 ) 来 构成 每 个 JMenu。 当 构造 完 一 个 JMenuBar 后 ， 可 以 使 用 
setJMenuBar() 方 法 把 它 安装 到 当前 程序 上 。 注 意 ， 当 按钮 按 下 的 时 候 ， 它 将 通过 调用 
getJMenuBar0 来 判断 当前 安装 的 是 哪 一 个 菜单 条 ， 然 后 换 成 另 一 个 菜单 条 。 

在 测试 Open 菜 单项 的 时 候 ， 要 注意 拼写 和 大 小 写 是 很 关键 的 ， 如 果 没 有 任何 匹配 的 Open， 
Java 也 不 会 报告 任何 错误 。 这 种 类 型 的 字符 串 比较 是 造成 程序 错误 的 根源 之 一 。 

菜单 项 的 选中 和 清理 能 够 被 自动 地 处 理 。 处 理 JCheckBoxMenuItem 的 代码 演示 了 判断 菜单 
项 是 否 被 选中 的 两 种 方式 : 字符 串 比较 (缺乏 安全 的 方式 ， 尽 管 你 会 看 到 可 以 使 用 这 种 方法 )， 
和 上 比较 事件 的 目标 对 象 。 可 以 使 用 getState0 方 法 得 到 是 否 选中 的 状态 。 还 可 以 用 setState0 方 法 
来 改变 JCheckBoxMenultem 的 状态 。 

菜单 对 应 的 事件 有 些 不 一 致 ， 这 可 能 会 引起 困惑 : JMenuItem 使 用 的 是 ActionListener， 而 
JCheckBoxMenuitem 使 用 的 是 ItemListener。JIMenu 对 象 虽然 也 支持 ActionListener， 不 过 其 用 
处 并 不 大 。 一 般 来 说 ， 要 把 监听 器 关联 到 每 一 个 JMenuItem、JCheckBoxMenuItem 或 者 








1356] 








JRadioButtonMenultem |, {BÆ ft, ItemListenerflActionListener% K] T REWA [357] 


组 件 上 。 

Swing 支持 助 记 键 ， 或 者 称 为 “键盘 快捷 键 "， 所 以 可 以 用 键盘 而 不 是 鼠标 来 选择 任何 从 
AbstractButton (按钮 ， 菜 单项 等 等 ) 继承 而 来 的 组 件 。 做 到 这 一 点 很 简单 ， 只 要 使 用 重 载 的 构 
造 器 ， 使 它 的 第 二 个 参数 接受 快捷 键 的 标识 符 即 可 。 不 过 ， 大 多 数 AbstractButton 没 有 这 样 的 构 
造 器 ， 所 以 更 通用 的 做 法 是 使 用 setMnemonic0 方 法 。 上 例 中 为 按钮 和 部 分 菜单 项 添加 了 快捷 
键 ， 快 捷 指示 符 会 自动 出 现在 组 件 上 。 

你 还 能 看 到 setActionCommand0 的 用 法 。 它 看 起 来 有 些 奇怪 ， 因 为 在 每 种 情况 下 , “动作 命 
令 ” 与 菜单 上 的 标签 都 完全 相同 。 为 什么 不 直接 使 用 标签 而 是 这 种 额外 的 字符 串 呢 ? 问题 在 于 
对 国际 化 的 支持 。 如 果 要 把 程序 以 另 一 种 语言 发 布 ， 最 好 是 希望 只 改变 菜单 上 的 标签 ， 而 不 用 
修改 代码 〈 毫 无 疑问 ， 修 改 代码 会 引入 新 的 错误 ) 。 通 过 使 用 setActionCommandO， 可 以 把 “ 动 
作 命令 ”作为 不 变量 ， 而 把 菜单 上 的 标签 作为 可 变量 。 所 有 的 代码 在 运行 时 都 使 用 “动作 命令 "， 
这 样 改变 菜单 标签 的 时 候 就 不 会 影响 代码 。 注 意 ， 在 本 例 中 ， 并 非 所 有 菜单 都 是 基于 “动作 命 
令 ” 进 行 判断 的 ， 这 是 因为 没有 专门 为 它们 设 定 “ 动 作 命 令 ”。 





~ 


1358} 
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大 量 工作 都 是 在 监听 器 中 完成 的 。BEL 执 行 的 是 JMenuBar 的 交换 。 在 ML 中 ， 采 用 了 “ 找 出 
按 铃 者 ”方式 ， 它 的 做 法 是 先 得 到 ActionEvent 的 事件 源 ， 然 后 把 它 类 型 转换 成 JMenuItem， 接 
着 得 到 其 “动作 命令 ”的 字符 串 ， 并 且 把 它 传递 给 级 联 的 诈 竹 句 进行 处 理 。 

尽管 FL 监听 器 处 理 的 是 风味 菜单 中 所 有 不 同 风味 的 菜单 项 ， 但 它 的 确 很 简单 。 如 果 事 件 处 
理 逻 辑 足够 简单 的 话 ， 这 种 方式 值得 参考 。 不 过 一 般 情况 下 会 采用 在 FooL、BarL 和 BazL 里面 
所 使 用 的 方式 ， 它 们 只 被 关联 到 一 个 菜单 项 ， 所 以 就 不 需要 进行 额外 的 判断 ， 因 为 你 明确 知道 
是 谁 调用 了 监听 器 。 尽 管 这 种 方式 产生 了 更 多 的 类 ， 但 是 类 内 部 的 代码 会 更 短 ， 整 个 处 理 过 程 
也 更 安全 。 

你 会 发 现 ， 有 关 菜 单 的 代码 很 快 就 变 得 宛 长 而 凌乱 ， 这 时 ， 使 用 GUI 构造 工具 才 是 明智 的 
选择 。 好 的 工具 还 可 以 对 菜单 进行 维护 。 

练习 19: (3) 修改 Menusjava， 在 菜单 上 使 用 单 选 按钮 而 不 是 复 选 框 。 

练习 20: (6) 创建 一 个 程序 ， 它 可 以 将 一 个 文本 文件 断 开 成 单词 ， 将 这 些 单词 分 布 到 菜单 和 
子 菜单 上 ， 作 为 它们 的 标签 。 


22.8.14 弹出 式 菜单 
要 实现 一 个 JPopupMenu， 最 直接 的 方法 就 是 创建 一 个 继承 自 MouseAdapter 的 内 部 类 ， 然 
后 对 每 个 希望 具有 弹出 式 行为 的 组 件 ， 都 添加 一 个 该 内 部 类 的 对 象 ; 


11: gui/Popup. java 

// Creating popup menus with Swing. 

import javax.swing.*; 

import java.awt.*; 

import java.awt.event.*; 

import static net.mindview.util. SwingConsole.*; 


public class Popup extends JFrame { 
private JPopupMenu popup = new JPopupMenu(); 
private JTextField t = new JTextField(16); 
public Popup() { 
setLayout (new FlowLayout()) ; 
add(t) ; 
ActionListener al = new ActionListener() { 
public void actionPerformed(ActionEvent e) { 
t.setText(((JMenuItem)e. getSource()).getText()); 


vi 
JMenultem m = new JMenuItem("Hi ther") ; 
m.addActionListener (al); 
popup. add (m) ; 
m = new JMenuItem("Yon"); 
m.addActionListener (al); 
popup. add (m) ; 
m= new JMenuItem("Afar"); 
m.addActionListener (al); 
popup. add (m) ; 
popup. addSeparator () ; 
m = new JMenuItem("Stay Here"); 
m.addActionListener (al); 
popup. add (m) ; 
PopupListener pl = new PopupListener(); 
addMouseListener (pl) ; 
t.addMouseListener(p1l); 

} 

class PopupListener extends HouseAdapter { 
public void mousePressed(MouseEvent e) { 

maybeShowPopup(e) ; 


} 
public void mouseReleased(MouseEvent e) { 


| ee 
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maybeShowPopup(e) ; 


private void maybeShowPopup(MouseEvent e) { 
if(e. isPopupTrigger()) 
popup. show(e.getComponent(), e.getX(), e.getY()); 


} 
public static void main(String[] args) { 
run(new Popup(), 309, 200); 
} 
} Ii~ 


同一 个 ActionListener 被 添加 到 了 每 一 个 JMenuItem 上 ， 它 要 从 菜单 标签 中 抓 取 文本 ， 然 后 
插入 到 JTextField 中 。 


22.8.15 绘图 

如 果 使 用 好 的 GUI 框 架 ， 绘 图 应 该 非常 简单 ，Swing 库 正 是 如 此 。 对 于 任何 绘图 程序 ， 问 题 
在 于 决定 绘图 位 置 的 计算 通常 比 对 绘图 功能 的 调用 要 复杂 得 多 ， 并 且 这 些 计算 程序 常常 与 绘图 
程序 混在 一 起 ， 所 以 看 起 来 程序 的 接口 比 实际 需要 的 要 更 复杂 。 

为 了 简化 问题 ， 考 虑 一 个 在 屏幕 上 表示 数据 的 问题 ， 在 这 里 ， 数 据 将 由 内 置 的 Math.sin0 方 
法 提供 ， 它 可 以 产生 数学 上 的 正弦 函数 。 为 了 使 事情 变 得 更 有 趣 一 些 ， 也 为 了 进一步 演示 Swing 
组 件 使 用 起 来 有 多 么 简单 ， 我 们 在 窗 体 底部 放置 了 一 个 滑 块 ， 用 来 动态 控制 所 显示 的 正弦 波 周 
期 的 个 数 。 此 外 ， 如 果 调 整 了 视窗 的 大 小 ， 你 会 发 现 正弦 波 能 够 自动 调整 ， 以 适应 新 的 视窗 。 

,尽管 在 任何 JComponent 上 都 可 以 绘图 ， 而 且 正 因为 如 此 ， 可 以 把 它们 当 作 画布 ， 但是， 要 
是 你 只 是 想 有 一 个 可 以 直接 绘图 的 平面 的 话 ， 典 型 的 做 法 是 从 JPanel 继 承 。 唯 一 需要 覆盖 的 方 
法 就 是 paintComponentO ， 在 组 件 必 须 被 重新 绘制 的 时 候 调 用 它 〈 通 常 不 必 为 此 担心 ， 因 为 何 
时 调用 由 Swing 决定 )。 当 此 方法 被 调用 时 ，Swing 将 传人 一 个 Graphics 对 象 ， 然 后 就 可 以 使 用 这 
个 对 象 绘图 了 ， 或 在 平面 上 绘制 了 。 

在 下 面 的 例子 中 ， 所 有 与 绘制 动作 相关 的 代码 都 在 SineDraw 类 中 ，SineWave 类 只 是 用 来 配 
置 程序 和 滑 块 控制 。 在 SineDraw 中 ，setCycles( 方 法 提供 了 一 个 钩子 hook) ， 它 允许 其 他 对 象 
(在 这 个 例子 中 就 是 滑 块 控制 ) 控制 周期 的 个 数 。 


/1/: gui/SineWave. java 
// Drawing with Swing, using a JSlider. 

import javax.swing.*; 

import javax.swing.event.*; 

import java.awt.*; 

import static net.mindview.uti1.SwingConsole.*; 


class SineDraw extends JPanel { 
private static final int SCALEFACTOR = 200; 
private int cycles; 
private int points; 
private double[] sines; 
private int[] pts; 
public SineDraw() { setCycles(5); } 
public void paintComponent (Graphics g) { 
super .paintComponent (g) ; 
int maxWidth = getWidth(); 
double hstep = (double)maxwidth / (double)points; 
int maxHeight = getHeight(); 
pts = new int{points}; 
for(int i = 0; i < points; i++) 
pts[1] = 
(int) (sines{i] * maxHeight/2 * .95 + maxHeight/2); 
g.setColor (Color . RED) ; 
for(int 1 = 1; i < points: i++) { 








1360] 
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int xl = (int)((i - 1) * hstep); 
int x2 = (int) (i * hstep); 
int yl = pts[i-1]; 
int y2 = pts[1]; 
g-drawLine(xl, yl, x2, y2); 
} 


} 
1361 public void setCycles(int newCycles) { 
cycles = newCycles: 
points = SCALEFACTOR * cycles * 2; 
sines = new double[points] ; 
for(int i = 0; i < points; i++) { 
double radians = (Math.PI / SCALEFACTOR) * i; 
sines[i] = Math.sin(radians); 
} 
repaint(); 
} 
} 


public class SineWave extends JFrame { 
private SineDraw sines = new SineDraw(); 
private JSlider adjustCycles = new JSlider(1, 30, 5); 
public SineWave() { 
add(sines) ; 
adjustCycles.addChangeListener(new ChangeListener() { 
public void stateChanged(ChangeEvent e) { 
sines.setCycles( 
((Slider)e. getSource()).getValue()); 
} 


ne 
add (BorderLayout.SOUTH, adjustCycles); 





} 
public static void main(String[] args) { 
run(new SineWave(), 760, 480); 


) 
} Min 
在 计算 正弦 波 上 的 点 的 过 程 中 用 到 了 所 有 的 字段 和 数组 ，cycles 表 示 所 希望 的 完整 的 正弦 波 
个 数 ，points 是 将 要 绘制 的 点 的 总 数 ，sines 包 含 了 正弦 函数 的 值 ，pts 包 含 将 要 绘制 在 JPanel 上 
点 的 ?坐标 。setCyeles( 方 法 先 根据 所 需 的 点 数 创建 数组 ， 然 后 为 数组 里 的 每 个 元 素 计算 相应 的 
正弦 函数 值 。 它 通过 调用 repaint0 方 法 ， 迫 使 调用 paintComponent0， 这 样 ， 余 下 的 计算 和 重 绘 
动作 就 会 发 生 。 
当 覆 盖 paintComponent0 方 法 的 时 候 ， 必 须 先 调用 该 方法 的 基 类 版 本 ， 然 后 才 可 以 做 想 做 
的 事情 ， 通常， 这 意味 着 要 使 用 你 在 java.awt.Graphics 的 文档 (在 JDK 文 档 中 ， 可 从 
1362] http://java.sun.com 找 到 ) 中 可 以 找到 的 Graphics 方 法 向 JPanel 上 绘制 像素 点 。 这 里 ， 几 乎 所 有 的 
代码 都 和 计算 有 关 ， 实 际 上 只 有 setColor0 和 drawLine0 这 两 个 方法 和 操纵 屏幕 有 关 。 在 创建 自 
己 的 用 于 显示 图 形 数据 的 程序 时 ， 可 能 会 有 类 似 的 经 历 : 把 大 部 分 时 间 用 在 所 绘 内 容 的 决定 上 ， 
而 真正 的 绘制 过 程 则 非常 简单 。 
在 我 创建 此 程序 的 时 候 ， 把 大 量 时 间 用 在 显示 正弦 波 上 。 做 完 这 些 之 后 ， 我 觉得 如 果 要 是 
能 够 动态 改变 周期 的 个 数 ， 那 么 它 的 效果 会 更 好 。 当 我 准备 这 么 做 的 时 候 ， 我 在 别 的 语言 中 的 
编程 经 验 使 我 觉得 ， 这 并 不 容易 实现 ， 但 是 结果 却 表明 ， 这 是 整个 程序 中 最 容易 的 部 分 。 我 先 
创建 了 一 个 JSlider (其 构造 器 参数 分 别 是 最 左边 的 值 、 最 右边 的 值 和 初始 值 ， 它 还 有 其 他 的 构 
造 器 ) ， 然 后 把 它 加 入 到 JApplet 中 。 接 下 来 ， 我 参考 了 JDK 文 档 ， 发 现 它 仅 有 的 监听 器 是 
addChangeListener， 它 在 滑 块 的 变动 足以 产生 新 值 的 时 候 被 触发 。 唯 一 能 够 处 理 此 事件 的 方法 
显然 就 是 stateChanged0， 它 提供 了 一 个 ChangeEvent 对 象 ， 这 样 就 可 以 追溯 到 变动 源 并 找 出 新 
值 。 通 过 调用 sines 对 象 的 setCycles0 就 可 采用 这 个 新 值 ，JPanel 也 将 被 重 绘 。 

















| 
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通常 ， 你 会 发 现在 Swing 程序 中 遇 到 的 问题 ， 大 部 分 都 可 以 通过 下 列 相似 的 过 程 得 到 解决 ， 
而 且 这 个 过 程 一 般 非 常 简单 ， 甚 至 从 未 使 用 过 组 件 也 是 如 此 。 

如 果 要 解决 更 复杂 的 问题 ， 可 以 使 用 更 高 级 的 绘图 库 ， 包 括 第 三 方 提供 的 JavaBeans 组 件 或 
者 Java 2D API。 这 些 内 容 超出 了 本 书 的 范围 ， 不 过 ， 要 是 你 的 绘图 代码 确实 变 得 过 于 复杂 了 ， 
你 就 应 该 查找 这 些 内 容 的 相关 资源 。 

练习 21， (5) 修改 SineWave.java， 加 入 相应 的 getter 和 setter 方 法 ， 使 SineDraw 成 为 一 个 
JavaBean, 

练习 22: (7) 使 用 SwingConsole 编 写 一 个 应 用 程序 。 它 有 三 个 滑 块 ， 每 一 个 分 别 表示 
java.awt.Color 类 型 的 红 、 绿 、 蓝 颜色 值 ， 窗 体 的 其 他 部 分 是 一 个 JPanel， 用 来 显示 由 三 个 滑 块 
所 决定 的 颜色 。 加 入 一 个 不 可 编辑 的 文本 域 ， 显 示 当 前 的 RGB 值 。 1363} 

练习 23: (8) 以 SineWave.java 为 参考 创建 一 个 程序 ， 它 可 以 在 屏幕 上 显示 旋转 的 正方 形 ， 
并 且 有 一 个 滑 块 可 以 控制 旋转 的 速度 ， 还 有 一 个 滑 块 可 以 控制 正方 形 的 尺寸 。 

练习 24: (7) 还 记得 “绘图 板 ” 这 个 玩具 吗 ? 它 有 两 个 调节 器 ， 一 个 用 来 控制 绘图 点 垂直 方 
向 的 运动 ， 一 个 用 来 控制 水 平方 向 的 运动 。 以 SineWavejava 程 序 为 基础 ， 编 写 一 个 具有 类 似 功 
能 的 程序 。 这 里 调节 器 可 以 使 用 请 块 来 实现 。 添 加 一 个 可 以 控 除 整个 图 形 的 按钮 。 

练习 25: (8) 在 SineWavejava 的 基础 上 编写 程序 (一 个 使 用 SwingConsole 类 的 应 用 程序 ) ， 
在 观察 窗口 画 一 条 动态 正弦 波 ， 它 可 以 像 示 波 器 那样 向 后 滚动 ， 使 用 一 个 线程 来 控制 动画 。 动 
画 的 速度 由 java.swingJSlider 控 件 进行 控制 。 

练习 26: (5) 修改 前 一 个 练习 ， 在 程序 里 创建 多 个 显示 正弦 波 的 面板 。 面 板 的 数目 可 以 通过 
HTML 标 记 或 命令 行 参数 进行 控制 。 

练习 27: (5) 修改 练习 25， 使 用 java.swing.Timer 类 来 控制 动画 。 注 意 它 与 java.utiTimer 类 
的 区 别 。 

练习 28:， (7) URARTE (只 是 一 个 类 ， 没 有 GUI) PROUT MRT FE AE AAR 
子 。 画 出 一 条 表示 每 次 掷 般 子 的 点 数 总 和 的 曲线 ， 然 后 在 你 气候 子 的 次 数 越 来 越 多 时 ， 动 态 地 
展开 显示 这 条 曲线 。 
22.8.16 对 话 框 

对 话 框 是 从 视窗 弹出 的 另 一 个 窗口 。 它 的 目的 是 处 理 一 些 具体 问题 ， 同 时 又 不 会 使 这 些 具 
体 细节 与 原先 窗口 的 内 容 混在 一 起 。 对 话 框 通常 应 用 于 视窗 编程 环境 中 。 

如 果 要 编写 一 个 对 话 框 ， 就 需要 从 JDialog 继 承 ， 它 只 不 过 是 另 一 种 类 型 的 Window， 与 
JFrame 类 似 。JDialog 具 有 一 个 布局 管理 器 (默认 情况 下 为 BorderLayout) ， 并 且 要 添加 事件 监 
听 器 来 处 理事 件 。 下 面 是 个 简单 的 例子 : 


/1: gui/Dialogs. java 
// Creating and using Dialog Boxes. 
import javax.swing.*; 1364) 
import java.awt.*; 

import java.awt.event.*; 

import static net.mindview.util. SwingConsole.*; 





























class MyDialog extends JDialog { 
public MyDialog(JFrame parent) { 
super(parent, "My dialog", true); 
setLayout(new FlowLayout()); 
add(new JLabel("Here is my dialog")); 
JButton ok = new JButton("0K") ; 
ok.addActionListener(new ActionListener() { 
| public void actionPerformed(ActionEvent e) { 
dispose(); // Closes the dialog 
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} 
H: 
add (ok); 
setSize(150,125); 


} 


public class Dialogs extends JFrame { 

private JButton bl = new JButton("Dialog Box"); 

private MyDialog dlg = new MyDialog(null); 

public Dialogs() { 

bl.addActionListener(new ActionListener() { 
public void actionPerformed(ActionEvent e) { 
dlg.setVisible(true) ; 

} 


D: 
add(b1); 

} 

public static void main(String{] args) { 
run(new Dialogs(), 125, 75); 





} 

YM» 

一 旦 创建 了 JDialog 以 后 ， 必 须 调用 setVisible(true) 方 法 来 显示 和 激活 它 。 当 对 话 框 被 关闭 
时 ， 你 必须 通过 调用 display0 来 释放 该 对 话 框 使 用 的 资源 。 

下 面 的 例子 更 加 复杂 ;对话 框 由 一 个 网 格 构成 (使 用 GridLayout) ， 并 且 添 加 了 一 种 特殊 按 
钮 ， 它 由 ToeButton 类 定义 。 按 钮 将 先 在 自己 周围 画 一 个 边框 ， 然 后 根据 状态 的 不 同 ， 在 中 央 显 
示 “ 空 白 "，"x” 或 者 “o"。 开 始 时 的 按钮 状态 为 “空白 "， 然 后 根据 每 一 轮 单 击 ， 变 成 “x” 或 
者 “o"。 而 且 ， 当 你 在 非 “空白 ”的 按钮 上 单 击 的 时 候 ， 它 将 在 “x” 和 “o” 之 间 翻 转 ， 以 提 
供 一 种 有 趣 翻转 (tic-tac-toe) 概念 的 变 体 。 此 外 ， 通 过 改变 在 主 应 用 视窗 中 的 数字 ， 可 
以 为 对 话 框 设置 任意 的 行 数 和 列 数 。 


11: gui/TicTacToe. java 
// Dialog boxes and creating your own components. 
import javax.swing.*; 

import java.awt.*; 

import java.awt.event.*; 

import static net.mindview.util.SwingConsole.*; 








public class TicTacToe extends JFrame { 
private JTextField 
rows = new JTextField("3"), 
cols = new JTextField("3"); 
private enum State { BLANK, XX, 00 } 
static class ToeDialog extends JDialog { 
private State turn = State.XX; // Start with x's turn 
ToeDialog(int cellsWide, int celisHigh) { 
setTitle("The game itself"); 
setLayout (new GridLayout (cellsWide, cellsHigh)); 
for(int i = @; i < cellsWide * cellsHigh; i++) 
add(new ToeButton()) : 
setSize(cellswide * 50, cellsHigh * 50); 
setDefaultCloseOperation(DISPOSE_ON_CLOSE) ; 


} 
class ToeButton extends JPanel { 
private State state = State. BLANK; 
public ToeButton() { addMouseListener(new ML()); } 
public void paintComponent (Graphics g) { 
super .paintComponent (g) ; 


getSize().width - 
getSize()-height - 
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yl = y2/4 
int wide = x2/2, high = y2/2; 
if (state == State.XX) { 

g.drawLine(x1, yl, x1 + wide, y1 + high): 

g.drawLine(x1. y1 + high, x1 + wide, yl): [1366] 
} 
if(state == State.00) 

gdrawOval (x1, yl, x1 + wide/2, y1 + nigh/2); 


class ML extends MouseAdapter { 
public void mousePressed(Mousetvent e) { 
if(state == State.BLANK) { 
state = turn; 
turn = 
(turn == State.Xxx ? State.00 : State.Xxx); 
} 
else 
state = 
(state == State.XX ? State.00 : State.XX); 
repaint(); 


è 
} 
} 
class BL implements ActionListener { 
public void actionPerformed(ActionEvent e) { 
JDialog d = new ToeDialog( 
new Integer (rows.getText()), 
new Integer (cols.getText())): 
d.setVisible(true); 
} 


} 

public TicTacToe() { 
JPanel p = new JPanel(); 
p.setLayout (new GridLayout(2,2)); 
p.add(new JLabel("Rows", JLabel.CENTER)); 
p. add (rows); 
p.add(new JLabel("Columns", JLabel.CENTER)); 
p.add(cols); 
add(p, BorderLayout . NORTH) ; 
JButton b = new JButton("go"); 
b.addActionListener (new BL()); 
add(b, BorderLayout . SOUTH) ; 


} 
public static void main(String{] args) { 
run(new TicTacToe(), 208, 200); 


} Via [1367] 
为 static 关 键 字 只 能 处 于 类 的 外 层 ， 所 以 内 部 类 不 能 包含 静态 的 数据 或 者 戏 套 类 。 

paintComponent( 方 法 先 在 面板 周围 绘制 正方 形 ， 然 后 在 中 间 画 “x” 或 “o"。 这 里 充满 了 
乏味 的 计算 ,但 是 却 很 直接 明了 。 

MouseListener 被 用 来 捕获 鼠标 单 击 事件 : 首先 ， 它 检查 面板 上 是 否 为 空白 , 如 果 不 是 空白 ， 
就 向 父 窗 体 查询 现在 是 哪 一 轮 ， 这 样 就 得 到 了 ToeButton 的 状态 。 通 过 内 部 类 机 制 ，ToeButton 
可 以 操作 其 外 部 类 ， 更 新 当前 的 轮 次 ， 如 果 按 钮 已 经 显示 为 “x” 或 “o"， 那 么 就 翻转 它 。 在 这 
个 计算 中 ， 读 者 可 以 看 到 第 3 章 学 习 的 if-else 三 元 运算 符 的 习惯 用 法 。 在 状态 改变 之 后 ， 
ToeButton 将 被 重 绘 。 

ToeDiaiog 的 构造 器 非常 简单 ， 它 按 你 的 要 求 向 网 格 中 添加 数 个 按钮 ， 然 后 调整 窗 体 大 小 ， 
使 得 每 个 按钮 的 长 和 宽 均 为 50 像 素 。 














Ba 
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TicTacToe 通 过 创建 两 个 JTextField (用 来 输入 按钮 网 格 的 行 数 和 列 数 ) 和 带 有 Action- 
Listener 的 “go” 按 钮 ， 完 成 整个 程序 的 设置 工作 。 当 按钮 被 按 下 时 ， 将 获取 JTextField 里 面 的 
数据 ， 因 为 它们 是 字符 串 ， 所 以 使 用 静态 的 IntegerparseInt0 方 法 把 它们 转换 成 整数 。 


22.8.17 文件 对 话 框 

某 些 操作 系统 具有 大 量 特殊 的 内 置 对 话 框 ， 它 们 可 以 处 理 诸如 选择 字体 、 颜 色 、 打 印 机 等 
操作 。 基 本 上 所 有 的 图 形 操作 系统 都 支持 打开 和 保存 文件 ， 所 以 Java 提 供 了 JFileChooser， 它 封 
装 了 这 些 操作 ， 使 文件 操作 变 得 更 加 方便 。 

下 面 的 程序 演练 了 两 个 JFileChooser 对 话 框 ， 一 个 用 来 打开 文件 ， 一 个 用 来 保存 文件 。 大 多 
数 代码 读者 现在 应 该 都 已 经 很 熟悉 了 ， 所 有 有 趣 的 行为 都 集中 在 处 理 两 个 按钮 单 击 事件 的 动作 
监听 器 中 : 


/1: gui/FileChooserTest. java 
// Demonstration of File dialog boxes. 
import javax.swing.*; 
import java.awt.*; 
import java.awt.event.*; 
import static net.mindview.util.SwingConsole.*; 
public class FileChooserTest extends JFrame { 
private JTextField 
fileName = new JTextField(), 
dir = new JTextField(); 
private JButton 
open = new JButton(“Open"), 
save = new JButton("Save"); 
public FileChooserTest() { 
JPanel p = new JPanel(); 
open.addAct ionListener(new OpenL()); 
p.add (open) ; 
save.addAct ionListener (new SaveL()); 
p.add(save) ; 
add(p, BorderLayout. SOUTH) ; 
dir.setEditable(false) ; 
fileName. setEditable(false) ; 
p = new JPanel(); 
p.setLayout (new GridLayout(2,1)); 
P.add( fileName) ; 
p.add (dir); 
add(p, BorderLayout . NORTH) ; 
} 
Class Openl implements ActionListener { 
public void actionPerformed(ActionEvent e) { 
JFileChooser c = new JFileChooser(); 
// Demonstrate “Open” dialog: 
int rVal = c.showOpenDialog(FileChooserTest. this); 
if(rVal == JFileChooser.APPROVE_OPTION) { 
fileName. setText(c.getSelectedFile().getName()); 
dir.setText(c.getCurrentDirectory().toString()): 
} 
if(rVal == JFileChooser.CANCEL_OPTION) { 
fileName. setText("You pressed cancel"); 
dir.setText(""); 
} 
} 
} 
class Savel implements ActionListener { 
public void act fonPerformed(ActionEvent e) { 
JFileChooser c = new JFileChooser(): 
// Demonstrate "Save" dialog: 
‘int rVal = c.showSaveDialog(FileChooserTest. this); 
if (rVal == JFileChooser.APPROVE_OPTION) { 
fileName. setText (c.getSelectedFile().getName()); 
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dir.setText(c.getCurrentDirectory().toString()); 


} 
if(rVal == JFileChooser.CANCEL_OPTION) { 
fileName. setText("You pressed cancel"); 
dir.setText(""); 
d 
} 
} 
public static void main(String[] args) { 
run(new FileChooserTest(), 258, 158); 


) 

} H~ 

注意 JFileChooser 的 使 用 有 很 多 种 变化 ， 包 括 使 用 过 滤器 来 缩小 可 供 选 择 的 文件 名 范围 等 。 

对 于 “打开 文件 ”对 话 框 ， 调 用 showOpenDialog0 方 法 ， 对 于 “保存 文件 ”对 话 框 ,调用 
showSaveDialog0。 这 些 命令 直到 对 话 框 关闭 的 时 候 才 会 返回 。 此 时 JFileChooser 对 象 仍旧 存在 ， 
所 以 能 够 从 中 读 取 数据 。getSelectedFile0 和 getCurrentDireetory0 这 两 种 方法 用 来 查询 操作 的 返 
回 结果 。 如 果 它 们 的 返回 为 空 ， 就 表示 用 户 已 经 取消 操作 并 关闭 了 对 话 框 。 

练习 29: (3) 在 javax.swing 的 JDK 文 档 中 ， 查 找 JColorChooser。 写 一 个 程序 ， 加 入 一 个 按 
钮 ， 它 可 以 弹出 用 来 选择 颜色 的 对 话 框 。 
22.8.18 Swing 组 件 上 的 HTML 

任何 能 接受 文本 的 组 件 都 可 以 接受 HTML 文 本 ， 且 能 根据 HTML 的 规则 来 重新 格式 化 文本 。 
也 就 是 说 ， 可 以 很 容易 地 在 Swing 组 件 上 加 入 漂亮 的 文本 。 例 如 : 


/1/; gui/HTMLButton. java 
// Putting HTML text on Swing components. 
import javax.swing.*; 

import java.awt.*; 

import java.awt.event.*; 

import static net.mindview.util.SwingConsole.*; 


public class HTMLButton extends JFrame { 
private JButton b = new JButton( 
“chtml><b><font size=#2>" + 
"<center>Hello!<br><i>Press me now!"): 
public HTMLButton() { 
b.addActionListener (new ActionListener() { 
public void actionPerformed(ActionEvent e) { 
add(new JLabel("<html>" + 
“<i><font size=+4>Kapow!")); 
// Force a re-layout to include the new label: 
validate(); 


H: 
setLayout (new FlowLayout()): 
add(b) ; 
} 
public static void main(String[] args) { 
run(new HTMLButton(), 208, 500); 


} 

} I~ 

必须 使 文本 以 “<html>” 标 记 开始 ， 然 后 就 可 以 使 用 普通 的 HTML 标 记 了 。 注 意 ， 不 会 强 
制 要 求 你 添加 普通 的 结束 标记 。 

ActionListener 将 一 个 新 的 JLabel 添 加 到 窗 体 中 ， 它 也 包含 了 HTML 文 本 。 不 过 ， 这 个 标签 
不 是 在 构造 过 程 中 添加 的 ， 所 以 你 必须 调用 容器 的 validate0 方 法 来 强制 对 组 件 进行 重新 布局 
(这 样 就 能 显示 该 新 标签 了 )。 

还 可 以 在 JTabbedPane、JMenulItem、JToolTip、JRadioButton 以 及 JCheckBox 中 使 用 
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HTMLX Æ. 
练习 30: 3) 编写 一 个 程序 ， 展 示 在 前 一 段 文字 中 所 列 的 所 有 组 件 上 如 何 使 用 HTML 文 本 。 


22.8.19 滑 块 与 进度 条 

滑 块 (已 经 在 SineWavejava 中 使 用 过 了 ) 能 令 用 户 通过 前 后 移动 滑 点 来 输入 数据 ， 在 某 些 
情况 下 这 显得 很 直观 (比如 音量 控制 )。 进 度 条 能 以 从 “ 空 ” 到 “ 满 ” 的 动态 方式 显示 数据 ， 这 
也 能 给 用 户 以 直观 的 感受 。 我 最 喜欢 的 例子 是 把 滑 块 与 进度 条 关联 在 一 起 ， 这 样 当 你 移动 滑 块 
的 时 候 ， 进 度 条 就 可 以 跟着 作 相 应 的 改变 。 下 面 的 示例 还 展示 了 ProgressMonitor， 这 是 一 种 功能 
更 完备 的 弹出 式 对 话 框 ; 


//: gui/Progress.java 

// Using sliders, progress bars and progress monitors. 
import javax.swing.*: 

import javax.swing.border.*: 

import javax.swing.event.*; 

import java.awt.*; 

import static net.mindview.util.SwingConsole.*; 


public class Progress extends JFrame { 
private JProgressBar pb = new JProgressBar(); 
private ProgressMonitor pm = new ProgressMonitor( 
this, "Monitoring Progress", "Test", @, 168); 
private JSlider sb = 
new JSlider(JSlider HORIZONTAL, @, 160, 68); 
public Progress() { 
setLayout (new GridLayout(2,1)); 
add (pb) ; 
pm. setProgress(@) ; 
pm. setMi11isToPopup( 1008) ; 
sb.setValue(@) ; 
sb. setPaintTicks (true) ; 
sb. setMajorTickSpacing (28) ; 
sb. setHinorTickSpacing(S); 
sb. setBorder(new TitledBorder ("Slide Me")); 
pb. setModel(sb.getModel()); // Share model 
add(sb); 
sb. addChangeListener (new ChangeListener() { 
public void stateChanged(ChangeEvent e) { 
pm. setProgress(sb.getValue()); 


D; 


} 
public static void main(String[] args) { 
run(new Progress(), 309, 200); 


} 

} :~ 

把 两 个 组 件 联系 到 一 起 的 关键 在 于 让 它们 共享 一 个 模型 ， 就 像 下 面 这 一 行 一 样 : 

pb. setModel (sb. getHodel()); 

当然 ， 也 可 以 使 用 监听 器 进行 控制 ， 不 过 在 简单 的 情况 下 这 种 方法 更 直接 。ProgressMonitor 
并 没有 模型 ， 因 此 需要 使 用 监听 器 方式 。 注 意 ，ProgressMonitor 只 能 向 前 移动 ， 并 且 一 旦 移动 到 
底 就 会 关闭 。 

JProgressBar 相 当 简 单 ， 而 JSlider 就 有 许多 可 选项 ， 比 如 放置 方向 、 大 小 标记 等 等 。 你 应 
该 能 够 注意 到 ， 添 加 一 个 带 标题 的 边框 是 多 么 地 直截了当 。 

练习 31: (8) 编写 一 个 “渐进 的 进度 表示 器 "， 当 接近 结束 的 时 候 ， 它 的 进度 越 来 越 慢 。 加 
入 一 些 随机 的 行为 ,使 它 能 够 不 时 地 表现 出 加 速 的 效果 。 

练习 32: (6) 修改 Progressjava， 不 要 共享 模型 (model) ， 而 是 使 用 一 个 监听 器 来 关联 滑 块 
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和 进度 条 。 
22.8.20 选择 外 观 

“可 插 拔 外 观 ” 使 你 的 程序 能 够 模仿 不 同 的 操作 系统 的 外 观 。 你 甚至 可 以 在 程序 运行 期 间 动 
态 改变 程序 的 外 观 。 不 过 ， 通 常 只 会 在 以 下 二 者 中 选择 一 个 : 要 么 选择 “ 跨 平台 ”的 外 观 〈 即 
Swing 的 “金属 ”外 观 ) ， 要 么 选择 程序 当前 所 在 系统 的 外 观 ， 这 样 你 的 Java 程 序 看 起 来 就 好 像 
是 为 该 系统 专门 设计 的 (在 大 多 数 情况 下 ， 这 的 确 是 最 好 的 选择 ， 而 且 可 以 避免 误导 用 户 )。 实 
现 这 两 种 行为 的 代码 都 很 简单 ， 不 过 你 要 确保 在 创建 任何 可 视 组 件 之 前 先 调用 这 些 代 码 ， 这 是 
因为 组 件 是 根据 当前 的 外 观 而 创建 的 ， 而 且 创 建 之 后 就 不 会 改变 (很 少 会 在 程序 运行 的 时 候 改 
变 程序 的 外 观 ， 这 个 过 程 相 当 复 杂 ， 这 方面 的 内 容 可 以 参考 专门 研究 Swing 的 书 )。 

实际 上 ， 如 果 要 使 用 跨 平 台 的 “金属 ”外 观 ( 它 是 Swing 程 序 的 特征 )， 那 么 什么 都 不 用 做 
因为 它 是 默认 外 观 。 不 过 要 想 使 用 当前 操作 系统 的 外 观 。 ， 那 么 加 入 下 列 代码 即 可 ， 一 般 把 这 
些 代码 添加 在 main0 的 开头 ， 至 少 也 要 在 添加 任何 组 件 之 前 ; 


try { 
UIManager. setLookAndFeel( 
UIManager . getSystemLookAndFeelClassName()); 
} catch(Exception e) { 
throw new RuntimeException(e) ; 
} 


在 catch 子 句 中 你 什么 也 不 用 做 ， 因 为 一 旦 你 的 设置 代码 失败 了 ，UIManager 默 认 将 设置 成 
跨 平台 的 外 观 。 不 过 在 调试 的 时 候 ， 异 常 非常 有 用 ， 所 以 你 也 许 希望 通过 catch 子 句 看 看 所 发 生 
的 问题 。 

下 面 的 程序 能 通过 命令 行 参数 选择 外 观 ， 这 里 选择 了 几 种 组 件 ， 演 示 了 它们 在 选择 不 同 的 
外 观 时 的 表现 : 


//: gui/LookAndFeel. java 

// Selecting different looks & feels. 

// {Args: motif) 

import javax.swing.*; 

import java.ant.*; 

import static net.mindview.util.SwingConsole.*; 


public class LookAndFeel extends JFrame { 

private String[] choices = 

"Eeny Meeny Minnie Mickey Moe Larry Curly".split(” "); 
private Component[] samples = { 

new JButton("JButton"), 

new JTextField("JTextField"), 

new JLabel("JLabel"), 

new JCheckBox("JCheckBox"), 

new JRadioButton("Radio"), 

new JComboBox (choices), 

new JList (choices), 


)} 
public LookAndFeel() { 
super ("Look And Feel"); 
setLayout(new FlowLayout()); 
for (Component component : samples) 
add (component) ; 


private static void usageError() { 
System. out .printin( 
"Usage: LookAndFeel [cross|system|motif]"); 
system.exit(1); 


O 你 可 能 会 对 Swing 呈现 机 制 是 否 能 够 怡 当地 处 理 你 的 操作 环境 产生 一 些 争论 。 
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} 
public static void main(String[] args) { 
if (args. length == 0) usageError(): 
if(args[9] .equals("cross")) { 
try { 
UIManager .setLookAndFeel (UIManager . 
getCrossPlat formLookAndFeelClassName()) ; 
} catch(Exception e) { 
e.printStackTrace() ; 


} 
} else if(args[®).equals("system")) { 
try { 
UIManager . setLookAndFeel (UIManager. 
getSystemLookAndFeelClassName()); 
} catch(Exception e) { 
e.printStackTrace() ; 


} 
} else if(args[6] .equals(*motif*)) { 
try { 
UIManager .setLookAndFeel (com. sun. java. "+ 
“swing. plaf motif Moti fLookAndFeel"): 
} catch(Exception e) { 
e.printStackTrace() ; 


} else usageError(); 

// Note the look & feel must be set before 
// any components are created. 

run(new LookAndfeel(), 300, 300); 


} 

} H~ 

可 以 观察 到 ， 一 种 选择 是 明确 地 使 用 字符 串 来 指定 外 观 ， 比 如 MotifLookAndFeel 外 观 。 而 
且 ， 只 有 这 个 外 观 和 默认 的 “金属 ”外 观 能 够 合法 地 在 所 有 平台 上 使 用 ， 尽 管 有 针对 Windows 
和 Macintosh 外 观 的 字符 串 ， 但 它们 只 能 在 各 自 的 平台 上 使 用 才 合 法 ( 当 在 这 些 平台 上 调用 
getSystemLookAndFeelClassName0 方 法 时 ， 可 以 得 到 相应 的 字符 串 )。 

自己 编写 一 种 外 观 也 是 可 以 的 ， 比 如 ， 当 你 为 某 个 公司 编写 框架 代码 时 ， 他 们 要 求 有 独特 
的 外 观 。 这 是 一 项 大 工程 ， 这 远 远 超出 了 本 书 的 范围 。( 事 实 上 ， 你 会 发 现 这 也 超出 了 许多 专业 
Swing 书 的 范围 ! ) 
22.8.21 树 、 表 格 和 剪贴 板 

你 可 以 在 www.MindView 上 的 本 章 补 充 材料 中 找到 关于 这 些 主题 的 简介 和 示例 。 


22.9 JNLP 与 Java Web Start 


出 于 安全 目的 ， 我 们 可 以 为 applet 签 名， 这 在 www.MindView.net 上 的 本 章 在 线 补充 材料 中 进 
行 了 介绍 。 经 过 签名 的 applet 功 能 强大 ,能够 有 效 取代 应 用 程序 ， 不 过 它们 只 能 在 浏览 器 中 运行 。 
这 就 需要 在 客户 机 上 运行 浏览 器 ， 从 而 增加 了 额外 的 开销 ， 同 时 ， 它 也 限制 了 applet 的 用 户 界面 ， 
常常 带 来 视觉 上 的 混乱 。 因 为 Web 浏 览 器 有 自己 的 菜单 和 工具 条 ， 它 们 会 显示 在 applet 的 上 方 e 。 

Java 网 络 发 布 协议 (Java Network Launch Protocol, JNLP) 在 保持 applet 优 点 的 前 提 下 ， 解 决 
了 这 个 问题 。 通 过 一 个 JNLP 程 序 ， 你 可 以 在 客户 机 上 下 载 和 安装 单机 版 的 Java 应 用 程序 。 你 可 
以 通过 命令 行 、 桌 面 图 标 ， 或 者 与 你 的 JNLP 实 现 一 起 安装 的 应 用 程序 管理 器 来 执行 这 个 程序 。 
应 用 程序 甚至 可 以 从 其 最 初 被 下 载 的 网 站 上 运行 。 

JNLP 应 用 程序 可 以 在 运行 时 从 因特网 上 动态 下 载 资源 ， 并 且 能 够 在 用 户 连接 到 因特网 的 情 
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况 下 ， 自 动 检查 程序 的 版 本 。 也 就 是 说 ， 它 能 把 applet 的 所 有 优点 和 应 用 程序 的 优点 结合 起 来 。 

与 applet 类 似 ， 客 户 机 系统 也 会 小 心 对 待 JNLP 应 用 程序 的 。JNLP 程 序 是 基于 Web 的 ， 很 容 
易 下 载 ， 但 它 也 可 能 是 恶意 程序 。 鉴 于 此 ，JNLP 应 用 程序 遵循 与 applet 一 样 的 沙 盒 安全 限制 。 与 
applet 类 似 ， 它 们 也 能 被 部 署 到 签名 的 JAR 文 件 中 ， 这 能 让 用 户 自行 选择 是 否 信任 签名 人 。 与 
applet 不 同 的 是 ， 如 果 它 们 被 部 署 到 未 签名 的 JAR 文 件 中 ， 它 们 通过 JNLP API 提 供 的 服务 仍然 能 
够 请 求 访问 客户 系统 上 的 特定 资源 在 程序 运行 期 间 ， 用 户 必 须 同意 此 请 求 )。 

因为 INLP 描 述 的 是 一 个 协议 ， 而 不 是 实现 ， 所 以 要 使 用 JINLP， 必 须要 有 一 个 实现 。Java 
Web Start (或 者 简称 为 JAWS)， 是 由 Sun 免 费 提供 的 官方 参考 实现 ， 并 且 作为 Java SE5 的 一 部 分 
而 发 布 的 。 如 果 把 它 用 于 开发 ， 那 么 就 要 确保 它 的 JAR 文 件 (javaws.jar) 处 于 类 路 径 中 ， 最 简 
单 的 解决 方案 是 将 javaws.jar 的 常规 Java 安 装 路 径 jre/lib 添 加 到 类 路 径 中 。 如 果 从 Web 服 务 器 上 部 
署 你 的 JNLP 应 用 程序 ， 必 须 确 保 服务 器 能 够 识别 MIME 类 型 application/x-java-jnlp-file。 如 果 你 
用 的 是 最 新 版 的 Tomcat 服 务 器 (http://jakarta.apache.org/tomcat) ， 那 么 它 已 经 预先 配置 好 了 。 如 
果 你 用 的 是 别 的 服务 器 ， 就 要 参考 一 下 用 户 指南 。 

创建 一 个 INLP 应 用 程序 并 不 困难 。 要 先 编写 一 个 标准 应 用 程序 ， 然 后 把 它 打包 到 一 个 JAR 
文件 中 。 这 时 ， 需 要 提供 一 个 启动 文件 一 它 是 一 个 简单 的 XML 文件 ， 用 来 告诉 客户 端 系统 下 
载 和 安装 这 个 程序 所 需 的 所 有 信息 。 如 果 你 不 准备 为 JAR 文 件 签名 ， 那 么 对 于 你 将 要 在 客户 机 
上 访问 的 每 种 类 型 的 资源 ， 必 须 使 用 JINLP API 提 供 的 服务 进行 访问 。 

下 面 是 FileChooser Testjava 的 一 种 变 体 ， 它 使 用 了 JNLP 服 务 来 打开 对 话 框 ， 所 以 这 个 类 可 
以 被 打包 进 未 经 签名 的 JAR 文 件 ， 然 后 作为 INLP 应 用 程序 部 署 。 


11: gui/jnip/JnipFileChooser. java 
// Opening files on a local machine with JNLP. 

// (Requires: javax.jnlp.FileOpenService; 

// You must have javaws.jar in your classpath) 

// To create the jnipfilechooser.jar file, do this: 
cd .. 

Md. 

// jar cvf gui/jntp/jnipfilechooser.jar gui/jnlp/*.class 
package gui. jntp; 

import javax.jnip.*; 

import javax.swin 
import java.awt.*; 
import java. vent.*; 
import java.io.*; 











public class JnlpFileChooser extends JFrame { 

private JTextField fileName = new JTextField(); 
private JButton 

open = new JButton("Open”), 

save = new JButton("Save"): 
private JEditorPane ep = new JEditorPane(); 
private JScrollPane jsp = new JScrollPane(): 
private FileContents fileContents; 
public JnipFileChooser() { 

JPanel p = new JPanel(); 

open.addActionListener(new Opent ()); 

p.add (open): 

save. addAct fonListener(new Savel ()); 

p.add(save); 

fsp.getViewport() .add(ep) ; 

add(jsp, BorderLayout .CENTER) ; 

add(p. BorderLayout. SOUTH) ; 

fileName. setEditable(false) ; 

p = new JPanel(); 

p.setLayout (new GridLayout(2,1)); 
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p.add (fileName) ; 

add(p, BorderLayout .NORTH) ; 
ep.setContentType("text"); 
save. setEnabled(false) ; 


} 
class Openl implements ActionListener { 
public void actionPerformed(ActionEvent e) { 
FileOpenService fs = null; 
try { 
fs = (FileOpenService)ServiceManager . lookup( 
“javax. jnip.FileOpenService"); 
} catch(UnavailableServiceException use) { 
throw new RuntimeException(use) ; 


} 
if(fs != null) { 
try { 
fileContents = fs.openFileDialog(".", 
new Stringl]{"txt", "**}); 
if(fileContents == nutt) 
return; 
fileName. setText(fileContents.getName()): 
ep.read(fileContents.getInputStream(), null); 
} catch(Exception exc) { 
throw new RuntimeException(exc); 
} 
save, setEnabled(true); 
} 
i 
class Savel implements ActionListener { 
public void actionPerformed(ActionEvent e) { 
FileSaveService fs = null; 
try ( 
fs = (FileSaveService)Servicedanager . lookup( 
"javax. jntp.FileSaveService") ; 
} catch(UnavailableServiceException use) { 
throw new RuntimeException(use) ; 





} 
if(fs t= null) { 
try { 
fileContents = fs.saveFileDialog(".", 
new String{] ("txt"), 
new ByteArrayInputStream( 
ep. getText().getBytes()), 
fileContents.getName()) ; 
if(fileContents == null) 
return; 
fileName. setText (fileContents.getName()) ; 
} catch(Exception exc) { 
throw new RuntimeException(exc); 





} 
} 
} 
} 
public static void main(String[] args) { 
JnlpFileChooser fc = new JnlpFileChooser(); 
fc. setSize(400, 300: 
fc. setVisible(true) ; 





} 
dM i~ 
注意 ，FileOpenService 和 FileCloseService 类 是 从 javaxjnlp 包 导入 的 ， 在 代码 中 没有 一 处 是 
JFileChooser 对 话 框 所 直接 引用 的 。 这 里 使 用 的 两 个 服务 必须 使 用 ServiceManager.lookup0 方 法 
进行 请 求 ， 客 户 系统 上 的 资源 只 能 通过 此 方法 返回 的 对 象 进行 访问 。 这 里 ， 客 户 系 统 中 的 文件 
通过 JNLP 提 供 的 FileContent 接 口 进行 读 写 ， 任 何 通过 诸如 File 或 FileReader 对 象 直接 访问 资源 的 
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企图 都 将 导致 抛 出 SecurityException 异 常 ， 这 与 在 未 经 签名 的 applet 中 执行 这 些 操作 得 到 的 结果 
相似 。 如 果 你 想 用 这 些 类 ， 却 不 愿 为 INLP 的 服务 接口 所 限制 ， 那 么 就 必须 对 JAR 文 件 签名 。 

在 JnipFile Chooser.java 中 ， 被 注释 的 jar 命 令 将 产生 必需 的 JAR 文 件 。 下 面 有 一 个 为 上 面 的 
例子 准备 的 合适 的 启动 文件 。 


11:! gui/jntp/fitechooser.jntp 
7xnl verstonet2.0% encodings W1E-8"7> 
<jnlp spec = 
Covebase=”tile:C:/AAA- T1J4/code/gui/jnip* 
href="filechooser.jnip"> 
<information> 
<title>FileChooser demo application</title> 
<vendor>Mindview Inc.</vendor> 
<description> 
Jnlp File chooser Application 
</description> 
<description kind="short"> 





Demonstrates opening, reading and writing a text file 
</description> 
<icon href="mindview.gif"/> 
<offline-allowed/> 
</information> 
<resources> 
<j2se version="1.3+" 
href="http: //java.sun.com/products/autod1/j2se"/> 
<jar href="jnipfilechooser.jar" download="eager"/> 
</resources> 
<application-dese 
main-class="gui, jnlp. JnlpFileChooser"/> 
</jnlp> 
Wh~ 


你 会 发 现在 (从 www.MindView.net 处 ) 下 载 的 本 书 源 代码 文件 中 ， 这 个 被 存 为 file chooser. 
jnlp 的 启动 文件 没有 第 一 行 和 最 后 一 行 ， 并 且 与 JAR 文 件 在 同一 个 目录 中 。 你 可 以 看 到 ， 这 是 一 
个 XML 文件 ， 包 含 了 一 个 “<jnlp>” 标 记 。 它 有 一 些 子 元 素 ， 其 涵义 大 多 不 言 自明 。 

jnlp 元 素 的 spec 属 性 可 以 告诉 客户 系统 此 JNLP 程 序 所 遵循 的 规范 的 版 本 。codebase 属 性 指向 
可 以 找到 启动 文件 和 资源 的 URL。 这 里 它 指向 的 是 本 地 机 器 上 的 目录 ， 这 是 一 个 不 错 的 测试 程 
序 的 方法 。 注 意 ， 你 需要 将 这 个 路 径 修改 为 表示 你 机 器 上 的 恰当 目录 ， 从 而 使 得 程序 可 以 成 功 
地 加 载 它 。href 属 性 必须 指定 这 个 启动 文件 的 名 称 。 

information 标 记 也 有 几 个 子 元 素 ， 它 们 提供 了 有 关 应 用 程序 的 信息 。 这 些 信息 将 用 于 Java 
Web Start 的 管理 控制 台 或 者 类 似 程序 它们 可 以 安装 JNLP 程 序 ， 并 且 可 以 让 用 户 从 命令 行 运行 
程序 ， 使 其 更 快捷 等 等 )。 

resources 标 记 与 HTML 文 件 中 的 applet 标 记功 能 很 相似 。j2se 子 元 素 指定 程序 运行 时 需要 的 
j2se 版 本 ，jar 子 元 素 的 href 属 性 指定 包含 .class 文 件 的 JAR 文 件 。jar 元 素 还 有 一 个 download 属 性 ， 
它 的 值 可 以 是 “eager” 或 者 “lazy”"， 这 可 以 告诉 INLP 的 实现 在 程序 运行 之 前 ， 是 否 需要 下 载 整 
个 JAR 文 件 。 

application-dese 属 性 能 告诉 JNLP 实 现 ，JAR 文 件 中 的 哪个 类 是 可 执行 的 类 ， 即 指定 程序 的 
A. 

jip 标 记 的 另 一 个 有 用 的 子 元 素 是 security 标 记 ， 这 里 并 没有 演示 它 。 下 面 是 一 个 security 标 
记 的 例子 : 

<security> 


<all-permissions/> 
<security/> 


[9 
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当 把 程序 部 署 在 一 个 已 签名 的 JAR 文 件 中 时 ， 就 要 使 用 security 标 记 。 上 面 的 例子 不 需要 这 
个 标记 ， 因 为 所 有 本 地 资源 都 是 通过 JNLP 服 务 进 行 访问 的 。 

还 有 其 他 一 些 标记 可 用 ， 其 细节 部 分 可 以 参考 规范 http://java.sun.com/products/javawebstart/ 
download-spec.html, 

要 想 启动 这 个 程序 ， 你 需要 一 个 下 载 页 面 ， 它 包含 了 一 个 链接 到 这 个 .jnlp 文 件 的 超 文本 链 
接 。 下 面 是 这 个 页 面 的 大 致 内 容 (没有 第 一 行 和 最 后 一 行 ): 


1/1:! gui/jnlp/filechooser. html 
<html> ` 

Follow the instructions in JnlpFileChooser.java to 
build jmlpfilechooser.jar, then: 

<a href="filechooser.jnip">click here</a> 

</html> 

Mh 


一 旦 程序 下 载 完毕 ， 就 可 以 使 用 管理 控制 台 对 它 进行 配置 。 如 果 在 Windows 系 统 上 使 用 Java 
Web Start， 将 提示 你 是 否 为 程序 创建 一 个 快捷 方式 以 供 下 次 使 用 。 这 个 行为 是 可 以 配置 的 。 

这 里 只 介绍 了 两 种 INLP 服 务 ， 当 前 版 本 的 JNLP 规 范 包含 了 七 种 服务 。 每 个 服务 都 被 设计 用 
来 执行 特定 的 任务 ， 比 如 打印 ， 以 及 和 剪贴 板 有 关 的 剪 切 和 粘贴 等 。 你 可 以 在 http://java.sun.com 
上 找到 更 多 的 信息 。 


22.10 Swing 与 并 发 


当 你 用 Swing 编程 时 ， 就 是 在 使 用 线程 。 在 本 章 开头 部 分 就 曾经 看 到 过 ， 当 时 你 学 习 到 所 有 
事物 都 应 该 通过 SwingUtilities.invokeLater0 提 交 Swing 事 件 分 发 线程 。 但 是 ， 不 用 显 式 地 创建 
Thread 对 象 这 一 事实 意味 着 多 线程 问题 可 能 会 让 你 大 吃 一 惊 ， 你 必须 牢记 存在 着 一 个 Swing 事件 
分 发 线程 ， 它 始终 在 那里 ， 通 过 从 事件 队列 中 拉 出 每 个 事件 并 依次 执行 它们 ， 来 处 理 所 有 的 
Swing 事件 。 牢 记事 件 分 发 线程 ， 将 有 助 于 确保 你 的 应 用 免 遭 死 锁 和 竞争 条 件 的 影响 。 

本 节 将 讲述 在 使 用 Swing 时 所 产生 的 多 线程 问题 。 


22.10.1 长 期 运行 的 任务 
在 使 用 图 形 化 用 户 界面 编程 时 最 容易 犯 的 错误 之 一 ， 就 是 意外 地 使 用 了 事件 分 发 线程 来 运 
行 长 任务 。 下 面 是 一 个 简单 的 示例 : 


41%: gui /LongRunningTask. java 

// A badly designed program. 

import javax.swing.*; 

import java.awt.*; 

import java.awt.event.*; 

import java.util.concurrent.*; 

import static net.mindview.util.SwingConsole.*; 


public class LongRunningTask extends JFrame { 

private JButton 

bl = new JButton(*Start Long Running Task"), 

b2 = new JButton("End Long Running Task"); 
public LongRunningTask() { 

bl.addActionListener(new ActionListener() { 

public void actionPerformed(ActionEvent evt) { 
try { 
TimeUnit . SECONDS. sleep(3); 

} catch(Interruptedexception e) { 
System.out.printin("Task interrupted"); 
return; 

) 

System.out.printin("Task completed"); 
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} 


ne 
b2.addActionListener(new ActionListener() { 
public void actionPerformed(ActionEvent evt) { 
// Interrupt yourself? 
Thread.currentThread(). interrupt (); 


} 
H: 
setLayout(new FlowLayout()); 
add(b1) ; 
add(b2) ; 
} 
public static void main(String[] args). { 
run(new LongRunningTask(), 200, 150): 


) 

} Mi~ 

当 按 下 bl 时 ， 事 件 分 发 线程 会 突然 被 占用 去 执行 这 个 长 期 运行 的 任务 。 此 时 你 会 看 到 ， 这 
个 按钮 甚至 不 会 马上 弹 起 来 ， 因 为 正常 情况 下 会 重 绘 屏幕 的 事件 分 发 线程 现在 处 于 忙 状 态 。 并 
且 你 也 不 能 做 其 他 任何 事 ， 例 如 按 下 b2， 因 为 这 个 程序 在 bl 的 任务 完成 ， 从 而 使 得 事件 分 发 线 
程 再 次 可 用 之 前 ， 是 不 会 做 出 响应 的 。b2 中 的 代码 是 一 种 有 缺陷 的 尝试 ， 试 图 通过 中 断 事件 分 
发 线程 来 解决 这 个 问题 。 ， 

解决 之 道 当然 是 在 单独 的 线程 中 执行 长 期 运行 的 任务 ， 下 面 是 使 用 了 单线 程 的 Executor， 
它 会 自动 将 待 处 理 的 任务 排队 ， 然 后 每 次 执行 其 中 的 一 个 


//: gui/InterruptableLongRunningTask. java 
// Long-running tasks in threads. 

import javax.swing.*; 
import java.awt.* 
import java.awt.event.*; 

import java.util.concurrent.*; A 
import static net.mindview.util.SwingConsole 











class Task implements Runnable { 
private static int counter = @; 
private final int id = counter++; 
public void run() { 

System.out.printin(this + " started"); 

try { 
TimeUnit. SECONDS. sleep(3); 

} catch(InterruptedException e) { ~ 
System.out.println(this + " interrupted"); 
return; 

} 

System.out.printin(this + * completed"); 

$ 
public String toString() { return "Task " + id; } 
public long id() { return id; } 

}; 


public class InterruptableLongRunningTask extends JFrame { 
private JButton 
bl = new JButton("Start Long Running Task"), 
b2 = new JButton("End Long Running Task"); 
ExecutorService executor = 
Executors .newSingleThreadExecutor () ; 
public InterruptableLongRunningTask() { 
bl.addActionListener(new ActionListener() { 
public void actionPerformed(ActionEvent e) { 
Task task = new Task(); 
executor .execute(task) ; 
System.out.printin(task + " added to the queue"): 
} 
Ð: 














1384) 








(1385) 





818 gn 





b2.addActionListener(new ActionListener() { 
public void actionPerformed(ActionEvent e) { 
executor .shutdownNow(); // Heavy-handed 
} 


H: 
setLayout(new FlowLayout()): 
add(b1) ; 
add(b2); 
} 
public static void main(String[] args) { 
run(new InterruptableLongRunningTask(), 200, 156); 
} 
} Mi~ 


这 个 程序 有 了 一 些 改进 ， 但 是 当 你 按 下 b2 时 ， 它 会 在 ExecutorService 上 调用 shutdown- 
Now0， 从 而 会 禁用 它 。 如 果 你 试图 添加 更 多 的 任务 ， 那 么 就 会 得 到 异常 。 因 此 ， 按 下 b2 会 使 程 
序 不 起 作用 。 我 们 想 要 的 是 在 关闭 当前 任务 (并 放弃 待 处 理 任务 ) 的 同时 ， 不 会 停止 任何 其 他 
的 事物 ， 在 第 20 章 中 描述 的 Java SE5 的 Callable/Future 机 制 正 是 我 们 所 需要 的 。 我 们 将 定义 一 个 
被 称 为 TaskManager 的 新 类 ， 它 包含 多 个 元 组 ， 这 些 元 组 持 有 表示 任务 的 Callable 和 从 Callable 
中 返回 的 Future。 必 须 使 用 元 组 的 原因 是 因为 它 使 得 我 们 可 以 跟踪 最 初 的 任务 ， 这 样 就 可 以 获 
取 从 Future 中 无 法 获得 的 额外 信息 。 下 面 是 示例 : 


//: net/mindview/util/TaskItem. java 

// A Future and the Callable that produced it. 
package net.mindview.util; 

import java.util.concurrent.*; 


public class TaskItem<R,C extends Callable<R>> { 
public final Future<R> future; 
public final C task; 
public TaskItem(Future<R> future, C task) { 
this. future = future; 
this. task = task; 


} Mine 
在 java.util.concurrent 类 库 中， 默认 情况 下 ， 经 由 Future 是 无 法 获得 任务 的 ， 因 为 在 你 从 


了 uture 获 得 结果 时 ， 任 务 不 必 仍旧 留存 。 这 里 ， 我 们 通过 把 任务 存储 起 来 ， 强 制 它 仍旧 留存 。 


TaskManager 被 放 到 了 net.mindview.util 中 ， 因 此 它 可 以 当 作 通 用 使 用 工具 来 使 用 : 


1/: net/mindview/util/TaskManager .java 
// Managing and executing a queue of tasks. 
package net .mindview.util; 
import java.util.concurrent.*; 
import java.util.*; 
public class TaskManager<R,C extends Callable<R>> 
extends ArrayList<TaskItemcR,C>> { 
private ExecutorService exec = 
Executors .newSingleThreadExecutor(); 
public void add(C task) { 
add(new TaskItem<R,C>(exec. submit (task), task)): 


public List<R> getResults() { 
Iterator<TaskItem<R,C>> items = iterator(); 
List<R> results = new ArrayList<R>(); 
while(items.hasNext()) { 
TaskItem<R,C> item = items.next(); 
if (item. future. isDone()) { 
try { 
results .add(item. future. get()); 
} catch(Exception e) { 
throw new RuntimeException(e); 
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} 
items. remove() ; 
} 
} 
return results; 
} 
public List<String> purge() { 
Iterator<TaskItem<R,C>> items = iterator(); 
List<String> results = new ArrayList<String>(); 
while(items.nasNext()) { 
TaskItemsR,C> item = items.next(); 
// Leave completed tasks for results reporting: 
if(!item. future. isDone()) { 
results.add("Cancelling ”+ item.task); 
item.future.cancel(true): // May interrupt 
items .remove() 1 
} 
} 
return results; 
} 
} Mi~ 


TaskManager 是 TaskItem 的 ArrayList， 它 还 包含 一 个 单线 程 的 Executor， 因 此 当 你 用 一 个 
Callable 来 调用 addO 时 ， 它 会 提交 该 Callable， 然 后 将 产生 的 Future 连 同 最 初 的 任务 一 起 存储 起 
来 。 在 这 种 方式 中 ， 如 果 需 要 用 任务 来 执行 某 些 事 ， 你 都 会 拥有 一 个 指向 该 任务 的 引用 。 作 为 
一 个 简单 示例 ， 在 purge0 中 使 用 的 是 任务 的 toString0 方 法 。 

现在 ， 它 可 以 用 在 我 们 的 示例 中 去 管理 长 期 运行 的 任务 了 : 


//: gui/InterruptableLongRunningCallable. java 
// Using Callables for long-running tasks. 
import javax. swing 
import java.awt. 
import java.awt.event.*; 
import java.util.concurrent.*; 
import net.mindview.util.*; 
import static net.mindview.util.SwingConsole 

















class CallableTask extends Task 
implements Callable<String> { 
public String call() { 
run(); 
return "Return value of " + this; 
} 
+ 


public class 
InterruptableLongRunningCallable extends JFrame { 
private JButton 
bl = new JButton("Start Long Running Task"), 
b2 = new JButton("End Long Running Task"), 
b3 = new JButton("Get results"); 
private TaskManager<String,CallableTask> manager = 
new TaskManager<String,CallableTask>(); 
public InterruptableLongRunningCallable() { 
bl.addActionListener(new ActionListener() { 
public void actionPerformed(ActionEvent e) { 
CallableTask task = new CallableTask(); 
manager . add (task) ; 
System.out.printin(task + * added to the queue"); 
$ 
p): 
b2.addActionListener(new ActionListener() { 
public void actionPerformed(ActionEvent e) { 
for(String result : manager .purge()) 
System.out.printin(result); 
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$ 
H: 
b3.addActionListener (new ActionListener() { 
public void actionPerformed(ActionEvent e) { 
// Sample call to a Task method: 
for (TaskItem<String.CallableTask> tt : 
manager) 
tt.task.id(); // No cast required 
for(String result : manager.getResults()) 
System.out.printin(result) ; 
} 
Ye 
setLayout (new FlowLayout()) ; 
add(b1); 
add(b2) ; 
add(b3) ; 


} 
Public static void main(String{] args) { 


run(new InterruptableLongRunningCallable(), 208, 150); 
) hie 
正如 你 所 看 见 的 ，CallableTask 确 实 执行 了 与 Task 相 同 的 操作 ， 只 是 它 返回 了 结果 ， 在 本 例 
中 返回 的 结果 是 一 个 标识 该 任务 的 String。 

被 称 为 SwingWorker (来 自 Sun 的 Web 站 点 ) 的 非 Swing 实用 工具 (不 是 标准 Java 发 布 版 本 的 
一 部 分 ) 和 Foxtrot (来 自 http://foxtrot.sourceforge.net) 都 是 专门 设计 用 来 解决 类 似 问 题 的 。 但 是 
到 本 书 撰写 时 为 止 ， 这 些 实用 工具 还 没有 做 出 修改 从 而 能 使 它们 利用 Java SE5 的 Callable/Future 
机 制 。 
为 最 终 用 户 提供 某 种 可 视线 索 ， 以 表示 任务 正在 运行 以 及 其 执行 进度 ， 经 常 是 一 种 非常 重 
要 的 工具 。 这 通常 可 以 使 用 JProgressBar 或 ProgressMonitor 来 实现 ， 下 面 的 示例 使 用 了 Progress- 
Monitor: 


//: gui/MonitoredLongRunningCal lable. java 

// Displaying task progress with ProgressMonitors. 
import javax. swing 
import java.awt.*; 
import java.awt.event.*; 

import java.util.concurrent.*; 

import net.mindview.util.*; 

import static net.mindview.util.SwingConsole.*; 





class MonitoredCallable implements Callable<String> { 


private static int counter = 0; 
private final int id = counter++; 
private final ProgressMonitor monitor; 
private final static int MAX = 8; 


public MonitoredCallable(ProgressMonitor monitor) { 


this.monitor = monitor; 
monitor. setNote(toString()); 
monitor.setMaximum(MAX - 1); 
monitor. setMillisToPopup(500) ; 


} 
public String call() { 
System.out.printin(this + " started"); 
try { 
for(int i = @; i < MAX; i++) { 
TimeUnit MILLISECONDS . sleep (509) ; 
if (monitor. isCanceled()) 
Thread. currentThread().interrupt(); 
final int progress = i; 
SwingUtilities. invokeLater( 
new Runnable() { 
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public void run() { 
monitor .setProgress (progress); 


} ` 
) 


} 
} catch(InterruptedException e) { 
monitor.close(); 
System.out.printin(this + " interrupted"); 
return "Result: " + this + " interrupted": 
} 
System.out.printin(this +" completed"); 
return "Result: ”+ this + " completed"; 


} 
public String toString() { return "Task " + id; } 


public class MonitoredLongRunningCallable extends JFrame { 
private JButton 
bl = new JButton("Start Long Running Task"), 
b2 = new JButton("End Long Running Task"), 
‘b3 = new JButton("Get results"); 
private TaskManager<String.MonitoredCallable> manager = 
new TaskManager<String,MonitoredCallable>() ; 
public MonitoredLongRunningCallable() { 
bl.addActionListener(new ActionListener() { 
public void actionPerformed(ActionEvent e) { 
MonitoredCallable task = new HonitoredCallable( 
new Progresstoni tor ( 
Moni toredLongRunningCallable. this, 
“Long-Running Task", "", @, @) 


) 
manager .add (task) ; 
System.out.println(task + " added to the queue"); 


$. 


H: 
b2.addActionListener(new ActionListener() { 
public void actionPerformed(ActionEvent e) { 
for (String result : manager .purge()) 
System. out .printIn(result) ; 
} 


Hi; 

b3. addActionListener(new ActionListener() { 
public void actionPerformed(ActionEvent e) { 

for(String result : manager.getResults()) 
System.out.printin(result); 

F 

D: 

setLayout(new FlowLayout()); 

add(b1); 

add (b2); 

add(b3); 


} 
public static void main(String[] args) { 
run(new MonitoredLongRunningCallable(), 200, 500); 


} i 
} Mi~ 


MonitoredCallable 的 构造 器 接收 一 个 ProgressMonitor 作 为 参数 ， 并 且 其 call0 方 法 会 每 半 秒 
钟 更 新 一 次 该 ProgressMonitor。 注 意 ，MonitoredCallable 是 一 个 单独 的 任务 ， 因 此 不 应 该 尝试 
着 直接 控制 UI， 这 样 就 需要 使 用 SwingUtilities.invokeLater0 来 向 monitor 提 交 进 度 变化 信息 。 
Sun 的 Swing 教程 (在 http://java.sun.com 上 ) 展示 了 另 一 种 可 选 的 方式 ， 即 使 用 Swing 的 Timer， 
它 可 以 检查 任务 的 状态 并 更 新 监视 器 。 

如 果 监 视 器 的 cancel 按 钮 被 按 下 ，monitorisCanceled0 方 法 将 返回 true。 这 里 ， 任 务 只 是 在 
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其 自身 的 线程 上 调用 了 interruptO ， 这 个 方法 将 使 任务 进入 eatch 子 句 ， 而 monitor 将 在 此 由 








close( 方 法 终结 。 

剩 下 的 代码 与 之 前 代码 是 一 样 的 ， 只 是 创建 ProgressMonitor 成 为 了 MonitoredLong- 
RunningCallable 构 造 器 的 一 部 分 。 

练习 33，(6) 修改 InterruptableLongRunningCallable.java， 使 其 并 行 而 不 是 按 顺 序 地 运行 
所 有 任务 。 


22.10.2 可 视 化 线程 机 制 

下 面 的 例子 使 一 个 实现 了 Runnable 接 口 的 Jpanel 类 可 以 在 其 自身 上 绘制 不 同 的 颜色 。 程 序 
设 定 为 从 命令 行 接受 参数 ， 以 决定 颜色 块 的 数目 和 颜色 变化 的 时 间 间 隔 (线程 休眠 的 时 间 )。 尝 
试用 不 同 的 值 运行 程序 ， 你 可 能 会 发 现 一 些 在 你 的 平台 之 上 的 多 线程 实现 的 有 趣 的 、 难 以 言 表 
的 特性 : 


//: gui/ColorBoxes.java 

// A visual demonstration of threading. 

import javax.swing.*; 

import java.awt.*; 

import java.util.concurrent.*; 

import java.util.*; 

import static net.mindview.util.SwingConsole.*; 


class CBox extends JPanel implements Runnable { 
private int pause; 
private static Random rand = new Random(); 
private Color color = new Color (8); 
public void paintComponent (Graphics g) { 
g.setColor (color) ; 
Dimension s = getSize(); 
g.fillRect(®, 6, s.width, s.height): 


} 

public CBox(int pause) { this.pause = pause; } 

public void run() { 

try 人 
while(!Thread. interrupted()) { 

color = new Color(rand.nextInt (OxFFFFFF)); 
repaint(); // Asynchronously request a paint() 
TimeUnit MILLISECONDS .sleep(pause) ; 


} 
} catch(InterruptedException e) { 
// Acceptable way to exit 
) 
} 
} 


public class ColorBoxes extends JFrame { 
private int grid = 12; 
private int pause = 50; 
private static ExecutorService exec = 
Executors .newCachedThreadPool (); 
public void setup() { 
setLayout(new GridLayout (grid, grid); 
for(int i = 0; i < grid * grid: i++) { 
CBox cb = new CBox (pause) ; 
add (cb) ; 
exec. execute (cb) ; 


} 


} 
public static void main(String[] args) { 
ColorBoxes boxes = new ColorBoxes(); 
if (args. length > @) 
boxes.grid = new Integer (args [0]) ; 
if(args.length > 1) 
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boxes.pause = new Integer (args{1]): 
boxes. setUp(); 
run(boxes, 500, 400); 


} 

} Mi~ 

ColorBoxes 设 置 为 使 用 GridLayout 布 局 管理 器 ， 从 而 得 到 二 维 的 网 格 单元 。 然 后 加 入 适当 
数目 的 CBox 对 象 来 填充 网 格 ， 并 为 这 些 对 象 传人 pause 值 。 在 main0 中 可 以 发 现 ，pause 和 grid 
都 有 默认 值 ， 通 过 命令 行 传人 的 参数 值 可 以 修改 这 些 值 。 

所 有 的 工作 由 CBox 完 成 。 它 从 JPanel 继 承 ， 并 且 实 现 了 Runnable 接 口 ， 所 以 每 个 JPanel 同 
时 也 是 一 个 独立 任务 。 这 些 任务 都 是 通过 线程 池 ExecutorService 来 驱动 的 。 

当前 单元 的 颜色 是 color， 这 些 颜 色 是 通过 使 用 Color 构 造 器 创建 的 ， 该 构造 器 接受 一 个 24 位 
的 数字 ， 在 本 例 中 ， 这 个 数字 是 随机 创建 的 。 

paintComponent() 方 法 相当 简单 ， 它 只 是 将 颜色 设置 为 color， 并 用 这 种 颜色 填充 整个 
JPanel, 

在 run() 方 法 中 ， 可 以 看 到 一 个 无 穷 循 环 ， 它 先 把 cColor 设 置 成 新 的 随机 颜色 ， 然 后 调用 
repaintO 进 行 显示 。 接 着 调用 sleepO 使 线程 休眠 一 段 时 间 ， 这 个 时 间 可 以 从 命令 行 指定 。 

在 run0 中 对 repaint0 的 调用 应 该 受到 检查 。 初 看 起 来 ， 就 像 是 我 们 在 创建 许多 线程 ， 共 中 
每 一 个 都 强制 进行 绘制 。 看 上 去 这 违反 了 应 该 只 向 事件 队列 提交 任务 的 原则 ， 但 是 ， 这 些 线程 
并 未 实际 修改 共享 资源 。 当 它们 调用 repaint0 时 ， 并 未 强制 在 这 一 时 刻 立即 进行 绘制 ， 而 只 是 
设置 了 一 个 “ 脏 标志 "， 表 示 当 下 一 次 事件 分 发 线程 准备 好 重 绘 时 ， 这 个 区 域 是 重 绘 的 备 选 元 素 
之 一 。 因 此 ， 这 个 程序 不 会 引起 Swing 的 多 线程 问题 。 

当 事 件 分 发 线程 实际 执行 paint0 时 ， 首 先 调用 paintComponentO ， 然 后 是 paintBorder0 和 
paintChildren0。 如 果 你 需要 在 导出 组 件 中 覆盖 paint0 ， 就 必须 牢记 调用 基 类 版 本 的 paint0， 以 
使 得 它 仍旧 可 以 执行 正确 的 行为 。 

这 个 设计 很 灵活 ， 并 且 线 程 与 每 个 JPanel 都 联系 到 了 一 起 ， 所 以 你 可 以 试 着 创建 任意 多 的 
线程 。( 事 实 上 ， 这 个 数目 受 你 的 JVM 所 能 自如 处 理 的 线程 数目 的 限制 ,) 这 个 程序 还 能 做 一 个 
有 趣 的 基准 测试 ， 因 为 可 以 针对 不 同 的 JVM 线 程 实现 以 及 不 同 的 平台 ， 用 它 来 演示 这 些 实现 的 
动态 性 能 以 及 行为 上 的 差异 。 

练习 34: (4) 修改 ColorBoxesjava， 让 数 个 闪烁 点 (星星 ) 穿 过 画布 ， 然 后 随机 地 变化 这 些 
“星星 ”的 颜色 。 


22.11 可 视 化 编程 与 JavaBean 


到 目前 为 止 ， 读 者 已 经 看 到 了 Java 在 编写 可 重用 代码 方面 所 具有 的 价值 。 类 是 “最 可 重用 ” 
的 代码 单元 ， 因 为 它 把 性 质 (FR) 和 行为 (方法 ) 聚合 成 一 个 单元 ， 所 以 它 既 可 以 被 直接 重 
用 ， 也 可 以 通过 组 合 或 继承 得 到 重用 。 

继承 和 多 态 是 面向 对 象 编程 的 关键 部 分 ， 不 过 在 大 多 数 情况 下 ， 当 把 程序 放 在 一 起 时 ， 人 
们 真正 希望 的 是 能 够 精确 地 满足 需求 的 组 件 。 人 们 希望 把 这 些 部 件 像 电 子 工程 师 把 芯片 放 在 电 
路 板 上 一 样 地 集成 到 自己 的 设计 中 。 看 来 ， 应 该 有 某 种 方法 能 加 速 这 种 “模块 化 装配 ”形式 的 
编程 。 

“可 视 化 编程 ”是 首先 成 功 的 方式 ， 而 且 非 常 成 功 。 首 先是 微软 公司 的 Visual BASIC (VB), 
接着 是 Borland 公 司 的 Delphi (JavaBeans 的 设计 主要 就 是 受到 了 它 的 启发 ) ， 它 属于 第 二 代 产 品 。 
伴随 着 这 些 编程 工具 ， 组 件 也 以 可 视 的 形式 表现 ， 因 为 它们 通常 表现 为 一 些 诸如 按钮 或 文本 域 
的 可 视 组 件 ， 所 以 这 种 可 视 化 很 有 意义 。 实 际 上 可 视 化 表示 通常 就 是 组 件 在 运行 的 程序 中 可 被 
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观察 到 的 方式 。 所 以 可 视 化 编程 的 部 分 过 程 就 包括 了 从 选用 区 选择 组 件 ， 然 后 把 它 拖 放 到 窗 体 
上 。 在 拖 放 的 时 候 ， 应 用 程序 构建 集成 开发 环境 (Integrated Development Environment, IDE) 会 
帮 着 生成 相应 的 代码 ， 这 些 代码 将 在 程序 实际 运行 的 时 候 创 建 相 应 的 组 件 。 

简单 地 把 组 件 拖 放 到 窗 体 上 一 般 还 不 足以 完成 程序 。 通 常 ， 还 必须 改变 组 件 的 特征 ， 比 如 
颜色 、 文 本 、 所 连接 的 数据 库 等 等 。 可 以 在 设计 时 被 修改 的 特征 被 称 为 属性 。 可 以 使 用 IDE 构 建 
器 工具 对 组 件 的 属性 进行 设置 ， 在 创建 程序 的 时 候 ， 这 些 配 置信 息 将 被 保存 ， 这 样 就 可 以 在 程 
序 运行 的 时 候 恢复 这 些 信息 。 

现在 读者 可 能 已 经 习惯 了 这 样 的 思想 ， 即 对 象 不 仅仅 包含 了 特征 ， 它 还 包括 一 组 行为 。 在 
设计 时 ， 可 视 化 组 件 的 行为 可 部 分 地 由 事件 表示 ， 意 思 是 “这 是 可 以 在 组 件 上 发 生 的 动作 ”。 通 
常 ， 通 过 为 事件 编写 代码 来 决定 当 事 件 发 生 时 该 如 何 处 理 。 

关键 在 于 ， IDE 构 建 工 具 能 够 使 用 反射 机 制 来 动态 地 向 组 件 查询 ， 以 找 出 组 件 具 有 的 属性 和 
支持 的 事件 。 一 旦 查询 完毕 ， 它 将 显示 属性 并 且 人 允许 你 进行 修改 〈 在 你 构建 程序 的 时 候 会 把 状 
态 保存 起 来 )， 此 外 它 还 能 显示 事件 。 通 常 ， 需 要 在 事件 上 做 出 类 似 于 双击 的 操作 ，IDE 构 建 工 
具 会 为 你 生成 一 个 与 此 事件 关联 的 代码 体 。 这 时 你 要 做 的 就 是 编写 事件 发 生 时 将 要 执行 的 代码 。 

所 有 这 些 加 起 来 就 是 IDE 构 建 工具 能 够 帮 你 完成 的 大 量 工作 。 这 样 ， 你 就 能 将 精力 集中 于 程 
序 的 外 观 和 功能 上 ， 然 后 依赖 IDE 构 建 工具 为 你 管理 相关 的 细节 。 可 视 化 编程 工具 如 此 成 功 的 原 
因 就 在 于 ， 它 们 极 大 地 加 速 了 程序 的 构建 过 程 (当然 包括 了 用 户 界面 部 分 ， 而 且 常 常 对 程序 其 
他 部 分 的 编写 也 有 帮助 ) 。 

22.11.1 JavaBean 是 什么 

在 剥 去 层 层 包 庄 之 后 ,组件 只 不 过 就 是 一 段 代码 ， 通 常 以 类 的 形式 出 现 。 对 于 组 件 来 说 ， 
关键 在 于 要 具有 “能 够 被 IDE 构 建 工具 侦 测 其 属性 和 事件 ”的 能 力 。 要 编写 一 个 VB 组 件 ， 程 
序 员 不 得 不 根据 某 些 固定 规则 编写 相当 复杂 的 代码 ， 以 暴露 出 组 件 的 属性 和 方法 〈 数 年 之 后 
这 已 经 变 得 容易 多 了 ) 。Delphi 作 为 第 二 代 可 视 化 编程 工具 ， 语 言 本 身 就 是 围绕 着 可 视 化 编 
程 而 设计 的 ， 所 以 编写 可 视 化 组 件 要 容易 很 多 。 然 而 ， 通 过 引入 JavaBean，Java 把 可 视 化 组 
件 的 创建 带 到 了 最 高 的 层次 ， 因 为 Bean 就 是 一 个 类 。 要 编写 一 个 Bean， 你 不 必 添 加 任何 特 
殊 代 码 或 者 使 用 任何 特殊 的 语言 功能 。 实际 上 你 唯一 要 做 的 就 是 , 对 方法 的 名 称 做 少许 修改 。 
通过 方法 的 名 称 就 能 告诉 应 用 程序 构建 工具 ， 这 是 一 个 属性 、 一 个 事件 ， 还 是 只 是 一 个 普通 
方法 。 

在 JDK 文 档 中 ， 命 名 规则 使 用 了 错误 的 术语 “设计 模式 ”(design pattern) 。 这 是 不 恰当 的 ， 
因为 设计 模式 〈 参 考 www.MindView.com 网 站 上 的 《Thinking in Patterns (with Java)》) 即使 不 与 
这 些 概念 混在 一 起 ， 其 本 身 也 具有 足够 的 挑战 性 。 这 不 是 设计 模式 ， 这 只 是 一 个 命名 规则 ， 而 
且 非 常 简单 : 

D 对 于 一 个 名 称 为 xxx 的 属性 ， 通 常 你 要 写 两 个 方法 : getXxx0 和 setXxx0。 任 何 浏览 这 些 
方法 的 工具 ， 都 会 把 get 或 set 后 面 的 第 一 个 字母 自动 转换 为 小 写 ， 以 产生 属性 名 。get 方 法 返回 的 
类 型 要 与 set 方 法 里 参数 的 类 型 相同 。 属 性 的 名 称 与 get 和 set 所 依据 的 类 型 毫 无 关系 。 

D 对 于 布尔 型 属性 ， 可 以 使 用 以 上 getinset 的 方式 ， 不 过 也 可 以 把 get 替 换 成 is。 

3) Bean 的 普通 方法 不 必 遵 循 以 上 的 命名 规则 ， 不 过 它们 必须 是 public 的 。 

4) 对 于 事件 ， 要 使 用 Swing 中 处 理 监 听 器 的 方式 。 这 与 前 面 所 见 到 的 完全 相同 一 一 
addBounceListener(BounceListener) 和 removeBounceListener(BounceListener) 用 来 处 理 
BounceEvent 事 件 。 大 多 数 情况 下 ， 内 置 的 事件 和 监听 器 就 能 够 满足 需求 了 ， 不 过 也 可 以 自己 编 
写 事件 和 监听 器 接口 。 
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我 们 可 以 使 用 这 些 规则 编写 一 个 简单 的 Bean: 


//: frogbean/Frog. java 
// A trivial JavaBean. 
package frogbean; 

import java.awt.*; 
import java.awt.event.*: 


class Spots {} 


public class Frog { 

private int jumps; 

private Color color; 

Private Spots spots; 

private boolean jmpr; 

public int getJumps() { return jumps; } 

public void setJumps(int newJumps) { 
jumps = newJumps; 

3 

public Color getColor() { return color; } 

public void setColor(Color newColor) { 
color = newColor; 

} 

public Spots getSpots() { return spots; } 

public void setSpots (Spots newSpots) { 
spots = newSpots; 


Public boolean isJumper() { return jmpr; } 

Public void setJumper(boolean j) { jmpr = j; } 

public void addActionListener (ActionListener 1) { 
Wane 


} 
public void removeActionListener (ActionListener 1) { 
MW vee 


} 
public void addKeyListener(Keylistener 1) { 
7VA 


} 
public void removeKeyListener(KeyListener 1) { 
Wiss 


} 

/1 An "ordinary" public method: 

public void croak() { 
System.out.println(*Ribbet!"); 


} 
} Mi~ 


首先 ， 你 可 以 发 现 这 只 是 一 个 类 。 通 常 ， 所 有 的 字段 都 是 私有 的 ， 只 能 通过 方法 和 属性 进行 
访问 。 根 据 命名 规则 ，Bean 的 属性 是 jumps、color、spots 和 jumper (注意 属性 名 称 第 一 个 字母 的 
大 小 写 变化 )。 对 于 前 三 个 属性 ， 其 名 称 与 其 内 部 标识 符 的 名 称 相同 ， 不 过 通过 jumper 你 会 发 现 ， 
属性 名 称 并 不 要 求 你 为 内 部 变量 使 用 任何 特定 的 标识 符 (或 者 ， 实 际 上 甚至 不 需要 有 任何 内 部 变 
量 与 属性 对 应 )。 

根据 add 和 remove 方 法 对 相关 监听 器 的 命名 可 以 看 出 ， 这 个 Bean 所 处 理 的 事件 是 Action- 
Event 和 KeyEvent。 最 后 ， 你 可 以 发 现 普通 方法 croak0 也 是 Bean 的 一 部 分 ， 这 只 是 因为 它 是 公 
共 方 法 ， 而 不 是 因为 它 遵循 了 任何 命名 规则 。 
22.11.2 使 用 Introspector 抽 取出 Beaninfo 

JavaBean 模 式 的 最 关键 部 分 之 一 ， 表 现在 当 你 从 选用 区 拖 动 一 个 Bean， 然 后 把 它 放置 到 窗 
体 上 的 时 候 。IDE 构 建 工具 必须 能 够 创建 这 个 Bean (如 果 有 默认 构造 器 就 可 以 创建 )， 然 后 在 不 
访问 Bean 的 源 代码 的 情况 下 抽取 出 所 有 必要 信息 ， 以 创建 属性 和 事件 处 理 器 的 列表 。 


[En 
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部 分 解决 方案 在 第 14 章 就 出 现 了 : Java 的 反射 机 制 能 发 现 未 知 类 的 所 有 方法 。 对 于 解决 
JavaBean 的 这 个 问题 ， 这 是 个 完美 的 方案 ， 你 不 用 像 其 他 可 视 化 编程 语言 那样 使 用 任何 语言 附 
加 的 关键 字 。 实 际 上 ，Java 语 言 里 加 入 反射 机 制 的 主要 原因 之 一 就 是 为 了 支持 JavaBean (尽管 反 
射 也 支持 对 象 序列 化 和 远程 方法 调用 )。 所 以 ， 你 也 许 会 认为 IDE 构 建 工 具 的 编写 者 将 使 用 反射 
来 抽取 Bean 的 方法 ， 然 后 在 方法 里 面 查找 出 Bean 的 属性 和 事件 。 

这 当然 是 可 行 的 ， 不 过 Java 的 设计 者 希望 提供 一 个 标准 工具 ， 不 仅 要 使 Bean 用 起 来 简单 ， 
而 且 对 于 创建 更 复杂 的 Bean 也 能 够 提供 一 个 标准 方法 。 这 个 工具 就 是 Introspector (内 省 器 ) 类 ， 
这 个 类 最 重要 的 就 是 静态 的 getBeanInfo0 方 法 。 向 这 个 方法 传递 一 个 Class 对 象 引用 ， 它 能 够 完 
全 侦 测 这 个 类 ， 然 后 返回 一 个 BeanInfo 对 象 ， 可 以 通过 这 个 对 象 得 到 Bean 的 属性 、 方 法 和 事件 。 

通常 ， 你 不 用 关心 这 些 问题 ， 也 许 能 直接 从 货架 上 获得 大 多 数 想 用 的 Bean， 而 不 必 人 知道 底 
层 的 细节 。 你 只 需 把 Bean 拖 动 到 窗 体 上 ， 配 置 它们 的 属性 ， 然 后 为 感 兴趣 的 事件 编写 处 理 程序 
即 可 。 不 过 ， 使 用 Introspector 来 显示 Bean 的 信息 是 个 值得 学 习 的 练习 。 下 面 就 是 这 个 工具 ， 


//: gui/BeanDumper . java 


// Introspecting a Bean. 


import javax. swin| 
import java.awt 
import java.a 
import java. bean: 
import java.lang.reflect.*; 

‘import static net.mindview.util.SwingConsole.*; 









public class BeanDumper extends JFrame { 
private JTextField query = new JTextField(28); 
private JTextArea results = new JTextArea(); 
public void print (String s) { results.append(s + "\n"); } 
public void dump(Class<?> bean) { 
results.setText(""); 
BeanInfo bi = null; 
try { 
bi = Introspector .getBeanInfo(bean, Object.class): 
} catch(IntrospectionException e) { 
print("Couldn't introspect " + bean.getName()); 
return: 
} 
for (PropertyDescriptor d: bi.getPropertyDescriptors()){ 
Class<?> p = d.getPropertyType(); 
if(p == null) continue; 
print("Property type:\n ”+ p.getName() + 
“Property name:\n ”+ d.getName()): 
Method readMethod = d.getReadMethod() ; 
if(readMethod != null) 
print("Read method:\n * + readMethod) ; 
Method writeMethod = d.getWriteMethod(); 
if(writeMethod != null) 
print("Write method:\n " + writeMethod) ; 
print(*= ee 入 于 











} 
print("Public methods:"); 
for (MethodDescriptor m : bi.getMethodDescriptors()) 
print(m.getMethod().toString()); 
noe”) s 





for (EventSetDescriptor e: bi.getEventSetDescriptors()){ 
print("Listener type:\n " + 
e.getListenerType().getName()); 
for(Method Im : e.getListenerMethods()) 
print("Listener method:\n ”+ lm.getName()); 
for (MethodDescriptor lmd : 
e.getListenerMethodDescriptors() ) 





图 形 化 周 户 界面 827 | 





print("Method descriptor:\n ”+ Imd.getMethod()); 
Method addListener= e.getAddListenerMethod(); 
print("Add Listener Method:\n ”+ addlistener); 
Method removeListener = e.getRemovelistenerMethod(); 
\n "+ removeListener); 








} 


class Dumper implements ActionListener { 
public void actionPerformed(ActionEvent e) { 
String name = query.getText(); 
Class<?> c = null; 
try { 
© = Class. forName(name) ; 
} catch(ClassNotFoundException ex) { 
results.setText("Couldn't find ”+ name); 
return; 


dump (c) ; 
} 


} 
public BeanDumper() { 
JPanel p = new JPanel(); 
p.setLayout (new FlowLayout()); 
p.add(new JLabel ("Qualified bean name:")); 
p.add (query) ; 
add(BorderLayout.NORTH, p); 
add(new JScrollPane(results)) ; 
Dumper dmpr = new Dumper (); 
query. addActionListener (dmpr) ; 
query. setText ("frogbean. Frog"); 
// Force evaluation 
dmpr.actionPerformed(new ActionEvent(dmpr, ©, "*)); 





totic static void main(String[] args) { 
run(new BeanDumper(), 600, 580); 

} Vie 

BeanDumperdump( 方 法 执行 了 所 有 工作 。 首 先 ， 它 试图 创建 一 个 BeanInfo 对 象 ， 成 功 的 
话 ， 就 调用 BeanInfo 的 方法 得 到 有 关 其 属性 、 方 法 和 事件 的 信息 。 你 会 发 现 Introspector. 
getBeanInfo( 方 法 有 第 二 个 参数 ， 它 用 来 告诉 Introspector 在 哪个 继承 层次 上 停止 查询 。 因 为 我 
们 不 关心 来 自 Object 的 方法 ， 所 以 这 里 的 参数 让 Introspector 在 解析 来 自 Object 的 所 有 方法 前 售 
止 查询 。 

对 于 属性 来 说 ，getPropertyDeseriptors0 返 回 类 型 为 PropertyDescriptor 的 数组 ， 你 可 以 针 
对 每 一 个 PropertyDeseriptor 都 调用 getPropertyTypeO 来 得 到 “通过 属性 方法 设置 和 返回 的 对 象 ” 
的 类 型 。 然 后 ， 针 对 每 个 属性 ， 你 可 以 通过 getName0 方 法 得 到 它 的 别名 (从 方法 名 中 抽取 )， 
通过 getReadMethod() 方 法 得 到 读 方法 ， 通 过 getWriteMethod( 方 法 得 到 写 方法 。 后 两 个 方法 返 
回 Method 对 象 ， 它 们 能 够 用 来 在 对 象 上 调用 相应 的 方法 (这 是 反射 的 一 部 分 ) 。 

对 于 公共 方法 (包括 属性 方法 )，getMethodDescriptors0 方 法 返回 类 型 为 MethodDescriptor 
的 数组 。 对 于 数组 的 每 个 元 素 ， 你 可 以 得 到 相关 联 的 Method 对 象 ， 并 显示 它们 的 名 称 。 

对 于 事件 ，getEventSetDescriptors() 方 法 返回 类 型 为 EventSetDescriptor (或 是 别 的 什么 ) 
的 数组 。 可 以 对 数组 中 的 每 一 个 元 素 进行 查询 ， 以 得 到 监听 器 的 类 型 、 监 听 器 类 的 方法 ， 以 及 
添加 和 移 除 监听 器 的 方法 。BeanDumper 程 序 显 示 了 所 有 这 些 信息 。 

在 启动 之 后 ， 程 序 强制 评估 frogbean Frog。 下 面 是 程序 输出 ， 这 里 移 除了 不 必要 的 额外 细节 ， 


Property type: 
Color 
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Property name: 
color 
Read method: 
public Color getColor() 
Write method: 
public void setColor (Color) 





Property typ 

boolean 
Property name: 

jumper 
Read method: 

public boolean isJumper() 
Write method: 

public void set Jumper (boolean) 
Property type: 

int 
Property name: 

jumps 
Read method: 

public int getJumps() 
Write method: 

public void setJumps(int) 





Property type: 
frogbean. Spots 
Property name: 
spots 
Read method: 
public frogbean.Spots getSpots() 
Write method: 
public void setSpots(frogbean. Spots) 








Public methods: 
public void setSpots(frogbean. Spots) 

public void setColor (Color) 

public void setJumps(int) 

public boolean isJumper() 

public frogbean.Spots getSpots() 

public void croak() 

public void addActionListener (ActionListener) 
public void addKeyL istener (KeyListener) 

Color getColor() 

void setJumper (boolean) 

int getJumps() 

void removeActionListener (ActionListener) 
void removeKeyL istener (KeyListener) 










Event support 
Listener type: 

KeyListener 
Listener method: 

keyPressed 
Listener method: 

keyReleased 
Listener method: 

keyTyped 
Method descriptor: 

public abstract void keyPressed(KeyEvent) 
Method descriptor: 

public abstract void keyReleased(KeyEvent) 
Method descriptor: 

public abstract void keyTyped(KeyEvent) 
Add Listener Method: 

public void addkeyListener(KeyListener) 
Remove Listener Method: 
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public void removeKeyListener (KeyListener) 






Listener tyi 
ActionListener 
Listener method: 
actionPerformed 
Method descriptor: 
public abstract void actionPerformed (ActionEvent) 
Add Listener Method: 
public void addActionListener (ActionListener) 
Remove Listener Method: 
public void removeActionListener (ActionListener) 





这 里 列 出 的 是 Introspector 能 够 观察 到 的 从 你 的 Bean 中 产生 的 BeanInfo 对 象 中 的 大 多 数 信 
息 。 可 以 观察 到 属性 的 类 型 和 名 称 相互 独立 。 注 意 属性 名 称 使 用 小 写字 母 (唯一 例外 的 情况 是 ， 
属性 名 称 以 连续 多 个 大 写字 母 开头 )。 记 住 ， 你 在 这 里 看 到 的 方法 名 称 (比如 读 和 写 方法 ) ， 是 
从 Method 对 象 中 得 到 的 ， 它 可 以 用 来 在 对 象 上 调用 相关 联 的 方法 。 

公共 方法 列表 中 既 包 括 了 那些 与 属性 或 事件 无 关 的 方法 ， 比 如 croak0， 也 包括 了 那些 与 属 
性 或 事件 有 关 的 方法 。 这 些 就 是 你 可 以 通过 编程 在 Bean 上 调用 的 所 有 方法 ， 并 且 ， 为 了 让 你 的 
工作 更 容易 ，IDE 构 建 工具 可 以 在 你 编写 方法 调用 的 时 候 ， 显 示 这 个 列表 。 

最 后 ， 你 可 以 发 现 所 有 事件 都 被 完全 地 解析 了 出 来 ， 包 括 相关 的 监听 器 、 它 的 方法 ， 以 及 
添加 和 移 除 监听 器 所 用 的 方法 。 基 本 上 ， 一 旦 你 获得 了 BeanInfo 对 象 ， 你 就 可 以 得 到 Bean 的 所 
有 重要 信息 。 你 还 能 够 调用 Bean 上 的 方法 ， 甚 至 在 除了 对 象 以 外 (这 里 又 是 反射 的 功能 ) 再 没 
有 其 他 任何 信息 的 情况 下 ， 也 能 够 这 么 做 。 

22.11.3 一 个 更 复杂 的 Bean 

下 面 的 例子 稍微 复杂 一 些 (虽然 它 并 不 是 很 重要 )。 它 是 一 个 JPanel， 当 和 鼠标 移动 的 时 候 ， 
可 以 在 鼠标 周围 绘制 小 回 圈 。 当 你 按 下 和 鼠标 ， 单 词 “Bang!” 将 出 现在 屏幕 的 中 央 ， 并 且 触 发 一 
个 动作 监听 器 。 

你 可 以 改变 的 属性 包括 : 圆 围 的 大 小 ， 当 按 下 鼠标 时 所 显示 的 单词 的 颜色 、 大 小 和 文本 。 
BangBean 类 还 具有 addActionListener0 和 removeActionListener0， 所 以 你 可 以 自己 编写 监听 器 
并 与 之 关联 ， 当 用 户 在 BangBean 上 单 击 的 时 候 ， 你 的 监听 器 就 会 被 触发 。 你 应 该 能 够 识别 它们 
支持 的 属性 和 事件 : 


//: bangbean/BangBean. java 
// A graphical Bean. 
package bangbean; 

import javax.swing.*; 
import java. aw! 

import java. aw! 
import java.io 
import java.utit.*; 





public class 
BangBean extends JPanel implements Serializable { 
private int xm, ym; 
private int cSize = 20; // Circle size 
private String text = "Bang!"; 
private int fontSize = 48; 
private Color tColor = Color.RED; 
private ActionListener actionListener: 
public BangBean() { 
addMouseListener(new ML()); 
addMouseMotionListener(new MML()): 
bi 








1.402 
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public int getCircleSize() { return cSize; } 
public void setCircleSize(int newSize) { 
cSize = newSize: 
} 
public String getBangText() { return text; } 
public void setBangText(String newText) { 
text = newText; 
} 
public int getFontSize() { return fontSize: } 
Public void setFontSize(int newSize) { 
fontSize = newSize; 
} 
public Color getTextColor() { return tColor; } 
public void setTextColor (Color newColor) { 
tColor = newColor; 
$ 
public void paintComponent (Graphics g) { 
super .paintComponent (g) ; 
g.setColor (Color .BLACK 
g.drawOval(xm - cSize/2, ym ~ cSize/2, cSize, cSize); 
} 
// This is a unicast Listener, which is 
77 the simplest form of Listener management: 
public void addActionListener (ActionListener 1) 
throws TooManyListeners€xception { 
if(actionListener != null) 
throw new TooħanyListenersException() 
actionListener = 1; 
} 
Public void removeActionListener (ActionListener 1) { 
actionListener = null; 
} 
class ML extends HouseAdapter { 
public void mousePressed(MouseEvent e) { 
Graphics g = getGraphics(); 
g.setColor(tColor); 
8. setFont ( 
new Font("TimesRoman”, Font.BOLD, fontSize)); 
int width = g.getFontMetrics().stringWidth (text); 
B.drawString(text, (getSize().width - width) /2, 
getSize() .height/2); 
g.dispose(); 
// Call the Listener's method: 
if(actionListener != null) 
actionListener act ionPerformed( 
new Act ionEvent (BangBean. this, 
ActionEvent .ACTION PERFORMED, null)); 





} 
} 
Class MML extends MouseMotionAdapter { 
public void mouseMoved(MouseEvent e) { 
xm = e@,getX(); 
ym = e.getY(); 
repaint (); 
} 
} 
public Dimension getPreferredSize() { 
return new Dimension(206, 268); 
} 
} Mi~ 


首先 注意 到 的 是 ，BangBean 实 现 了 Serializable 接 口 。 这 就 意味 着 [DE 构建 工具 能 够 在 程序 
设计 者 调整 属性 之 后 ， 通 过 对 象 序列 化 机 制 “保存 ”(pickle) BangBean 的 所 有 信息 。 在 作为 运 
行程 序 的 组 件 创建 这 个 Bean 的 时 候 ， 这 些 保存 过 的 属性 能 够 被 恢复 ， 这 样 你 就 得 到 了 与 设计 时 








1405] 一 致 的 Bean。 
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观察 addActionListener() 方 法 的 特征 签名 ， 就 会 发 现 它 可 以 抛 出 TooManyListeners- 
Exception 异 常 。 这 表明 它 只 支持 单 路 的 事件 处 理 方式 ， 即 事件 发 生 的 时 候 它 只 能 通知 一 个 监听 
器 。 通常， 你 会 使 用 支持 多 路 方式 的 组 件 ， 这 样 一 个 事件 可 以 通知 给 多 个 监听 器 。 不 过 ， 这 样 
会 牵涉 到 线程 问题 ， 我 们 将 在 14.14.4 节 研究 这 个 问题 。 同 时 ， 单 路 事件 处 理 方式 就 可 以 回避 这 
个 问题 。 

当 单 击 鼠标 时 ， 文 字 将 显示 在 BangBean 的 中 央 ， 此 时 如 果 actionListener 字 段 非 空 ， 它 的 
actionPerformed() 方 法 将 被 调用 ， 调 用 之 前 要 创建 一 个 新 的 ActionEvent 对 象 。 当 鼠标 移动 的 时 
候 ， 新 的 坐标 将 被 捕获 ， 并 且 重 新 绘制 窗 体 (如 你 所 见 ， 擦 除 窗 体 上 的 任何 文字 ) 。 

下 面 是 测试 Bean 的 BangBeanTest 类 : 


//: bangbean/BangBeanTest. java 
// {Timeout: 5} Abort after 5 seconds when testing 
package bangbean: 
‘import javax.swing.*; 
import java.awt.*; 
import java.awt.event.*; 
import java.util.*; 
import static net.mindview.util.SwingConsole.*; 








public class BangBeanTest extends JFrame { 
private JTextField txt = new JTextField(28); 
// During testing, report actions: 
class BBL implements ActionListener { 
private int count = @; 
public void actionPerformed(ActionEvent e) { 
txt.setText("BangBean action "+ count++); 


} 
public BangBeanTest() { 
BangBean bb = new BangBean(); 
try { 
bb.addActionListener(new BBL()); 
} catch (TooManyListenersException e) { 
txt.setText("Too many Listeners"); 


} 

add (bb) ; 

add (BorderLayout.SOUTH, txt); 
} 


Public static void main(String{] args) { 
run(new BangBeanTest(), 400, 500); 


} 
} Mi~ 


当 Bean 处 于 IDE 中 时 ， 不 必 使 用 这 个 类 ， 不 过 为 你 的 每 个 Bean 提 供 一 种 快速 的 测试 方法 会 
很 有 帮助 。BangBeanTest 将 把 一 个 BangBean 对 象 放 在 applet 内 ， 给 BangBean 对 象 关 联 一 个 简单 
的 ActionListener (动作 监听 器 ) ， 当 ActionEvent 事 件 发 生 的 时 候 ， 监 听 器 能 把 事件 计数 显示 到 
JTextField。 当 然 ， 通 常 应 用 程序 物 建 工 具 会 帮助 创建 使 用 Bean 的 大 部 分 代码 。 

当 通 过 BeanDumper 查 询 BangBean， 或 者 把 BangBean 放 置 在 支持 Bean 的 开发 环境 中 时 ， 显 
示 出 的 属性 和 动作 将 比 上 面 代码 中 编写 的 要 多 许多 。 这 是 因为 BangBean 继 承 自 JPanel， 而 
JPanel 也 是 一 个 Bean， 所 以 你 会 看 到 所 有 的 属性 和 事件 。 

练习 35: (6) 在 因特网 上 查找 并 下 载 几 个 免费 的 GUI 构建 工具 ， 或 者 如 果 你 有 的 话 就 使 用 商 
业 产 品 ， 研 究 一 下 要 把 BangBean 导 人 这 个 环境 并 使 用 它 ， 需 要 哪些 必要 的 步骤 。 

22.11.4 JavaBean 与 同步 
当 创建 Bean 的 时 候 ， 必 须要 假设 它 可 能 会 在 多 线程 环境 下 运行 。 也 就 是 说 ， 


to 








Tog 
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1) 尽 可 能 地 让 Bean 中 的 所 有 公共 方法 都 是 synchronized (同步 ) 的 。 当 然 ， 这 将 导致 
synchronized 的 运行 时 开销 (在 近 几 个 版 本 的 JDK 中 ， 这 个 开销 已 经 大 大 降低 了 )。 如 果 这 么 做 
会 有 问题 ， 那 么 对 那些 不 会 导致 临界 区 域 问题 的 方法 ， 可 以 考虑 不 同步 。 但 要 记 住 ， 这 些 方法 
并 非 总 是 这 么 容易 做 出 判断 。 进 行 同步 的 方法 应 该 尽 可 能 短 (比如 下 面 例子 中 的 getCircleSize0 
Wek), 并且 (RE) 是 “原子 的 ”一 一 原子 性 是 指 ， 在 调用 含有 这 一 小 段 代码 的 方法 时 ， 对 象 
不 能 被 改变 (但 是 回顾 一 下 第 21 章 ， 你 认为 是 原子 性 的 代码 也 许 并 不 具有 原子 性 )。 不 同步 这 样 
的 方法 也 许 不 会 对 程序 的 执行 速度 有 明显 的 效果 。 所 以 最 好 是 同步 Bean 的 所 有 公共 方法 ， 只 有 
在 确实 会 有 明显 效果 并 且 你 可 以 安全 地 移 除 时 ， 才 在 一 个 方法 上 移 除 synchronized 关 键 字 。 

2) 当 一 个 多 路 事件 触发 了 一 组 对 该 事件 感 兴趣 的 监听 器 时 ， 你 必须 假定 ， 在 你 遍历 列表 进 
行 通知 的 同时 ， 监 听 器 可 能 会 被 添加 或 移 除 。 

第 一 点 很 容易 处 理 ， 但 第 二 点 就 需要 做 更 多 的 思考 。 前 面 版 本 的 BangBean.java 通 过 忽略 
Synchronized 关 键 字 并 使 用 单 路 事件 处 理 方式 ， 回 避 了 并 发 问题 。 下 面 是 修改 后 的 版 本 ， 它 可 以 
在 多 线程 环境 下 工作 ， 而 且 使 用 了 多 路 事件 处 理 方式 : 


//: gui/BangBean2. java 

// You should write your Beans this way so they 
// can run in a multithreaded environment. 
import javax.swing.*; 

import java.awt. 
‘import 
import 
import 
import static net.mindview.util.SwingConsole.*; 












public class BangBean2 extends JPanel 
implements Serializable { 
private int xm, ym; 
private int cSize = 20; // Circle size 
private String text = "Bang!"; 
private int fontSize = 48; 
private Color tColor = Color.RED; 
private ArrayList<ActionListener> actionListeners = y 
new ArrayList<ActionListener>(); 
public BangBean2() { 
addMouseListener(new HL()); 
addMouseMot ion istener (new HM()): 
} 
public synchronized int getCircleSize() { return cSize; } 
public synchronized void setCircleSize(int newSize) { 
cSize = newSize; 


public synchronized String getBangText() { return text; } 

public synchronized void setBangText(String newText) { 
text = newText: 

} 

public synchronized int getfontSize(){ return fontSize: } 

public synchronized void setFontSize(int newSize) { 
fontSize = newSize; 

} $ 


public synchronized Color getTextColor(){ return tColér:} 
public synchronized void setTextColor(Color newColor) { 
tColor = newColor; 


} 
public void paintComponent (Graphics g) { 
super .paintComponent (g) ; 
g.setColor (Color .BLACK) ; 
B-drawOval (xm - cSize/2, ym - cSize/2, cSize, cSize): 


$ 
// This is a multicast listener, which is more typically 
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JJ used than the unicast approach taken in BangBean. java: 
public synchronized void 
addActionListener (ActionListener 1) { 
actionListeners.add(1); 
} 
public synchronized void 
removeActionListener (ActionListener 1) { 
actionListeners.remove(1); 
} 
// Notice this isn’t synchronized: 
public void notifyListeners() { 
ActionEvent a = new ActionEvent (BangBean2. this, 
ActionEvent ACTION PERFORMED, null); 
ArrayList<ActionListener> lv = null; 
// Make a shallow copy of the List in case 
// someone adds a listener while we're 
V/ calling Listeners: 
synchronized(this) { 
lv = new ArrayList<ActionListener>(actionListeners): 
} 
// Call all the listener methods: 
for (ActionListener al-: 1v) 
al.actionPerformed(a) 
M 


class ML extends MouseAdapter { 
Public void mousePresséd(MouseEvent e) { 
Graphics g = getGraphics(); 
g-setColor(tColor); 
g.setFont( 
new Font("TimesRoman", Font.BOLD, fontSize)): 
int width = g.getFontMetrics().stringWidth(text) ; 
B-drawString(text, (getSize().width - width) /2, 
getSize() .height/2); 
g-dispose(); 
notifyListeners(); 
} 





class MM extends MouseMotionAdapter { 
public void mouseMoved(MouseEvent e) { 
xm = e.getX(); 
ym = e.getY(); 
repaint(); 
} 


} 
public static void main(String[] args) { 
BangBean2 bb2 = new BangBean2(); 
bb2.addActionListener(new ActionListener() { 
public void actionPerformed(ActionEvent e) { 
System.out.printin("ActionEvent" + è); 
} 


bb2.addActionListener (new ActionListener() { 
public void actionPerformed(ActionEvent e) { 
System. out.printin("BangBean2 action"); 





d 
Ye 
bb2. addActionListener(new ActionListener() { 
public void actionPerformed(ActionEvent e) { 
System.out.printin("More action"); 
} 


Ds; 
JFrame frame = new JFrame(); 
frame. add(bb2) ; 
run(frame, 366, 360); 
$ 
dM 
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给 方法 加 上 synchronized 关 键 字 很 容易 。 不 过 要 注意 ， 在 addActionListener0 和 remove- 
ActionListener( 方 法 中 ， 现 在 要 从 ArrayList 添 加 或 移 除 ActionListener， 所 以 你 可 以 添加 任意 多 
的 监听 器 。 

可 以 观察 到 notifyListeners0 方 法 没有 被 同步 。 这 个 方法 可 以 同时 被 多 个 线程 调用 。 在 调用 
notifyListeners0 期 间 ， 也 可 能 有 别 的 线程 在 对 addActionListener0 或 removeActionListener0 进 
行 调 用 ， 这 将 遍历 actionListeners 的 这 个 ArrayList， 所 以 就 可 能 发 生 冲 突 。 为 了 解决 这 个 问题 ， 
先 在 一 个 同步 子 句 里 对 ArrayList 进 行 复制 ， 可 以 通过 使 用 ArrayList 的 构造 器 来 实现 ， 因 为 它 会 
复制 其 参数 中 的 元 素 ， 然 后 对 复制 的 对 象 进行 遍 历 。 这 样 ， 就 可 以 操作 原来 那个 ArrayListi 不 
会 对 notifyListeners0 过 程 有 影响 了 。 

paintComponent() 方 法 也 没有 被 同步 。 判 断 是 否 同步 覆盖 后 的 方法 并 不 像 判 断 自 己 写 的 方 
法 那么 明显 。 在 本 例 中 表明 ， 无 论 是 否 同步 ，paintComponent0 方 法 看 起 来 工作 都 正常 。 不 过 
你 必须 考虑 这 些 问题 : 

1) 这 个 方法 会 修改 对 象 中 “关键 ”变量 的 状态 吗 ?要 和 弄 清楚 变量 是 否 “关键 "， 必 须 判断 它 
们 是 否 被 程序 中 的 其 他 线程 读 写 。( 在 本 例 中 ， 读 写 操作 基本 上 是 通过 同步 方法 进行 的 ， 所 以 你 
检查 这 些 就 可 以 了 。) 在 paintComponent0 方 法 中 ， 就 没有 进行 任何 修改 操作 。 

2) 这 个 方法 依赖 于 那些 “关键 ”变量 吗 ? 如 果 有 某 个 同步 方法 会 修改 此 方法 所 使 用 的 变量 ， 
那么 你 应 该 把 这 个 方法 也 同步 。 根 据 这 一 点 ， 你 会 发 现 cSize 被 同步 方法 所 改变 ， 所 以 
paintComponent(0 方 法 应 该 被 同步 。 不 过 ， 这 里 还 可 以 问 自己 ,“ 如 果 cSize 在 调用 paintComponent0 
的 过 程 中 被 改变 ， 最 坏 的 结果 是 什么 ? ”要 是 觉得 问题 不 大 ， 这 种 改变 只 起 瞬时 作用 ， 你 就 可 以 
作出 不 同步 paintComponent0 方 法 的 决定 ， 以 避免 同步 方法 调用 所 产生 的 额外 开销 。 

3) 第 三 个 线索 是 查看 基 类 版 本 的 paintComponent0 是 否 同步 ， 在 这 里 并 没有 被 同步 。 这 不 
是 种 严密 的 判断 方法 ， 只 是 一 个 线索 。 比 如 在 本 例 中 ， 由 同步 方法 所 改变 的 字段 (比如 eSize) 
已 经 混在 paintComponent0 的 公式 中 了 ， 在 这 种 情况 下 它 有 可 能 会 被 改变 。 不 过 ， 请 注意 ， 同 
步 不 会 继承 ， 也 就 是 说 ， 如 果 基 类 方法 是 同步 的 ， 派 生 类 中 和 覆盖 后 的 版 本 并 非 自 动 同步 。 

4) paintO0 和 paintComponent0 的 执行 必须 尽 可 能 快 。 要 尽量 把 处 理 的 开销 移 到 方法 外 面 ， 
所 以 要 是 发 现 需要 同步 这 些 方法 ， 那 么 你 的 设计 可 能 就 存在 问题 。 

与 BangBeanTest 相 比 ，main0 里 面 的 测试 代码 已 经 被 修改 过 了 ， 它 通过 添加 类 外 的 监听 器 ， 
来 演示 BangBean2 的 多 路 事件 处 理 能 力 。 

22.11.5 把 Bean 打 包 

在 把 JavaBean 加 入 到 某 个 支持 Bean 的 IDE 之 前 ， 必 须 把 它 置 于 一 个 Bean 容 器 中 ，Bean 容 器 ， 
也 就 是 一 个 JAR 文 件 ， 它 里 面包 含 了 Bean 的 所 有 .class 文件 以 及 能 表明 “这 是 一 个 Bean” 的 “ 清 
单 ”(manifest) 文件 。 RE ERR ENE 对 于 BangBean， 它 的 清 
单 文件 看 起 来 像 这 样 : 


Manifest-Version: 1.0 





Name: bangbean/BangBean.class 
Java-Bean: True 


第 一 行 显示 了 清单 文件 格式 的 版 本 ， 目 前 Sun 公 司 发 布 的 是 1.0。 第 二 行 (忽略 空 行 ) 为 
BangBean.class 文 件 命名 ， 第 三 行 表明 “这 是 一 个 Bean" 。 没 有 第 三 行 ， 程 序 构建 工具 将 不 能 把 
类 识别 成 Bean。 

唯一 需要 技巧 的 部 分 是 ， 要 确保 在 “Name:” 字 段 写 上 正确 的 路 径 。 回 过 头 看 看 前 面 的 
BangBean.java， 就 会 发 现 它 处 于 bangbean 包 中 (也 就 是 一 个 名 为 “bangbean” 的 子 目录 ， 它 不 
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包括 在 classpath 中 ) ， 清 单 文件 的 名 称 字段 必须 含有 这 个 包 信息 。 此 外 ， 必 须 把 清单 文件 放 在 包 
的 根 目 录 的 上 层 目 录 中 ， 这 里 就 是 把 清单 文件 放 在 “bangbean” 目 录 的 父 目录 中 。 然 后 需要 在 
清单 文件 所 在 的 目录 下 调用 jar 工 具 ， 如 下 所 示 : 

Jar cfm BangBean.jar BangBean.mf bangbean 

这 里 假定 你 要 把 生成 的 JAR 文 件 命名 为 BangBean.jar， 并 且 清 单 文件 的 名 称 为 BangBean.mf。 

你 可 能 会 奇怪 :“ 当 我 编译 完 BangBean.java 之 后 ， 生 成 的 其 他 .class 文 件 在 哪 呢 ? ”其 实 ， 
它们 都 在 bangbean 子 目录 下 ， 上 面 jar 命 令 行 的 最 后 一 个 参数 就 是 bangbean 目 录 名 。 当 你 把 目录 
名 传递 给 jar 工 具 时 ， 它 将 把 整个 目录 打包 进 JAR 文 件 (在 本 例 中 , 包括 了 BangBeanjava 源 文件 ， 
你 也 许 不 会 选择 在 自己 的 Bean 中 包括 源 代码 )。 此 外 ， 如 果 你 把 刚才 生成 的 JAR 文 件 解 包 ， 就 会 
发 现 里 面 并 没有 你 指定 的 清单 文件 ，jar 工 具 创 建 了 自己 的 清单 文件 (部 分 根据 你 提供 的 信息 ) 
MANIFEST.MF， 这 个 文件 放 在 “META-INF”( 元 信息 ) 目录 下 。 打 开 这 个 文件 ， 就 会 看 到 
jar 为 每 个 文件 都 添加 了 数字 签名 信息 ， De 


Digest-Algorithms: SHA MDS 
SHA-Digest: pDpEAGQNaeCx8aF tqPI4udSX/00= 
MDS-Digest: O4NcS1hE3Smnz1p2hj 6qgeg== 


通常 ， 你 不 必 考虑 这 些 ， 如 果 改 变 了 程序 ， 你 只 要 修改 原来 的 清单 文件 ， 然后 重新 调用 jar 
工具 来 为 Bean 创 建 一 个 新 的 JAR 文 件 即 可 。 通 过 把 相关 信息 加 入 清单 文件 ， 你 还 能 把 其 他 Bean 
也 添加 到 这 个 JAR 文 件 中 。 

要 注意 的 一 点 是 ， 你 可 能 希望 把 每 个 Bean 都 放 进 专门 的 子 目录 中 ， 因 为 在 创建 JAR 文 件 的 
时 候 ， 你 把 子 目录 的 名 称 传递 给 了 jar 工 具 ， 它 将 把 子 目 录 里 面 的 所 有 文件 都 打包 进 JAR 文 件 。 
可 以 看 到 Frog 和 BangBean 都 在 它们 各 自 的 子 目录 中 。 

一 旦 把 Bean 正 确 地 打包 成 JAR 文 件 ， 就 可 以 把 它 导 入 支持 Bean 的 IDE 中 了 。 导 入 的 方式 根据 
不 同 的 工具 可 能 会 有 所 不 同 ， 不 过 Sun 公 司 在 它们 的 “Bean Builder” 里 提供 了 一 个 免费 使 用 的 
测试 工具 (可 以 从 http://java.sun.com/beans 下 载 )。 只 要 把 JAR 文 件 复制 到 正确 的 目录 下 ， 就 可 以 
把 你 的 Bean 导 入 到 Bean Builder 中 。 

练习 36: (4) 把 Frog.class 加 入 本 章 所 示 的 清单 文件 ， 执 行 jar 工 具 创 建 包含 Frog 和 BangBean 
的 JAR 文 件 。 然 后 从 Sun 下 载 并 安装 Bean Builder， 或 者 使 用 现 有 的 支持 Bean 的 程序 构建 工具 ， 把 
JAR 文 件 导 人 开发 环境 测试 这 两 个 Bean。 

练习 37，(5) 自己 编写 一 个 JavaBean， 取 名 为 Valve。 它 有 两 个 属性 : 一 个 布尔 型 的 on， 一 个 
整 型 的 level。 写 一 个 清单 文件 ， 使 用 jar 为 你 的 Bean 打 包 ， 在 Bean Builder 或 者 支持 Bean 的 程序 构 
建 工 具 里 面 导入 这 个 Bean， 然 后 进行 测试 。 
22,11.6 对 Bean 更 高 级 的 支持 

你 已 经 看 到 了 制作 一 个 Bean 是 多 么 简单 ， 不 过 并 不 局 限于 目前 所 看 到 的 功能 。JavaBean 架 
构 提供 的 门槛 很 低 ， 但 也 可 以 扩展 到 更 复杂 的 情况 。 这 些 情 形 超出 了 本 书 的 范围 ， 不 过 这 里 可 
以 做 一 些 简要 介绍 。 读 者 可 以 在 java.sun.com/beans 找 到 更 多 的 细节 。 

你 可 以 针对 属性 提供 高 级 功能 。 前 面 的 例子 只 演示 了 单一 属性 ， 但 也 可 以 使 用 数组 来 表示 
多 重 属性 。 这 称 为 索引 属性 。 只 要 提供 恰当 的 方法 (也 就 是 根据 命名 规则 给 方法 命名 ) ， 
Introspector 将 识别 出 索引 属性 ， 这 样 你 的 应 用 程序 构建 工具 就 可 以 正确 工作 。 

属性 可 以 被 绑 定 ， 即 它们 能 通过 PropertyChangeEvent 事 件 通知 其 他 对 象 。 这 些 被 通知 的 对 
象 可 以 根据 Bean 上 的 变化 来 决定 如 何 改变 自己 。 

属性 可 以 被 约束 ， 即 如 果 属 性 的 改变 是 不 可 接受 的 ， 其 他 对 象 可 以 否决 这 个 改变 。 这 些 对 
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象 也 是 通过 PropertyChangeEvent 事 件 得 到 通知 的 ， 而 且 能 抛 出 PropertyVetoException 异 常 来 阻 
止 属性 的 改变 ， 然 后 恢复 属性 的 旧 值 。 

还 可 以 改变 Bean 在 设计 阶段 的 表示 方式 : 

1) 可 以 为 自己 的 Bean 提 供 一 个 自 定义 的 属性 表 。 对 于 所 有 其 他 Bean， 将 使 用 通常 的 属性 
表 * 但 当 你 的 Bean 被 选中 的 时 候 ， 将 自动 激活 你 提供 的 表 。 

2) 还 可 以 为 特定 的 属性 提供 自 定义 的 编辑 器 ， 对 于 其 他 属性 ， 将 使 用 普通 编辑 器 ， 但 是 当 
你 的 特殊 属性 被 编辑 的 时 候 ， 将 自动 激活 你 提供 的 编辑 器 

3) 可 以 为 你 的 Bean 提 供 一 个 自 定义 的 BeanInfo 类 , 它 可 以 产生 与 默认 情况 下 由 Introspector 所 
提供 的 信息 不 同 的 信息 。 

4) 还 可 以 把 每 一 个 FeatureDescriptor 里 的 “专家 ”模式 打开 或 者 关闭 ， 这 样 就 能 够 区 分 简 
单 功能 和 复杂 功能 。 
22.11.7 有 关 Bean 的 其 他 读物 

有 很 多 关于 JavaBean 的 书籍 ， 比 如 《JavaBeans 》，Elliotte Rusty Harold 著 (IDG 出 版 ，1998) 。 


22.12 Swing 的 可 替代 选择 


尽管 Swing 类 库 是 Sun 支 持 的 GUI， 但 它 决 不 是 创建 图 形 化 用 户 界面 的 唯一 方式 。 有 两 种 重 
要 的 可 替代 选择 ， 即 用 于 Web 之 上 的 客户 端 GUI 的 使 用 MacroMedia 的 Flex 编 程 系 统 的 MacroMedia 
Rlash， 以 及 用 于 桌面 应 用 的 开源 的 Eclipse 标准 工具 包 (Standard Widget Toolkit, SWT) 类 库 。 

为 什么 要 考虑 可 替代 选择 呢 ? 对 于 Web 客 户 端 来 说 ， 你 可 以 有 相当 强 的 理由 ， 因 为 applet 失 
败 了 。 想 想 看 ， 它 们 已 经 存在 多 久 了 (从 Java 诞 生 时 就 有 了 ) ， 那 些 关于 applet 的 最 初 的 虚假 宣传 
和 和 承诺 又 存在 多 久 了 ， 现 在 偶尔 遇 到 使 用 applet 的 Web 应 用 仍旧 会 仍然 大 吃 一 惊 。 甚 至 Sun 都 没 
有 到 处 使 用 applet， 下 面 是 一 个 示例 : 

http://java.sun.com/developer/onlineTraining/new2java/javamap/intro. html 

在 Sun 的 网 站 上 ，Java 特 性 的 交互 地 图 看 起 来 很 像 是 一 个 Java applet， 但 实际 他 们 是 用 Flash 
实现 它 的 。 这 好 像 是 一 种 默认 ， applet 没 有 获得 成 功 。 更 重要 的 是 ，Flash Player 在 超过 98% 的 计 
算 平台 上 都 安装 了 ， 因 此 它 可 以 被 认为 是 一 种 已 经 被 接受 了 的 标准 。 如 你 所 见 ，Flex 系 统 提供 
了 非常 强大 的 客户 端 编程 环境 ， 肯 定 比 JavaScript 更 强大 ， 并 且 它 的 外 观 通常 要 优 于 applet。 如 果 
你 想 要 使 用 applet， 那 仍旧 必须 确保 客户 端 会 下 载 JRE， 但 是 比较 而 言 ，Flash Player 更 小 ， 下 载 
起 来 更 快 。 

对 于 桌面 应 用 , 使 用 Swing 的 一 个 问题 是 用 户 会 注意 到 他 们 在 使 用 完全 不 同 的 一 种 应 用 程序 ， 
因为 Swing 应 用 程序 的 外 观 与 一 般 的 桌面 应 用 程序 不 同 。 用 户 通常 不 会 对 应 用 程序 中 的 新 外 观感 
兴趣 ， 他 们 只 关注 完成 工作 ， 并 且 喜 欢 应 用 程序 的 外 观 与 他 们 所 有 其 他 的 应 用 程序 相同 。SWT 
创建 的 应 用 程序 看 上 去 和 本 地 应 用 程序 一 样 ， 因 为 它 的 类 库 尽 可 能 多 地 使 用 本 地 组 件 ， 这 些 应 
用 程序 运行 起 来 比 等 效 的 Swing 应 用 程序 要 快 。 


22.13 用 Flex 构 建 Flash Web 客 户 端 


因为 轻 量 级 的 MacroMedia Flash 虚 拟 机 非常 普遍 ,因此 大 多 数 人 都 可 以 使 用 基于 Flash 的 界面 ， 
而 无 需 安装 任何 东西 ， 并 且 它 的 外 观 和 行为 在 所 有 系统 和 平台 之 上 都 是 相同 的 。 
通过 使 用 Macromedia Flash， 你 可 以 为 Java 应 用 开发 Flash 用 户 界面 。Flex 由 基于 XML 和 脚本 
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的 编程 模型 以 及 非常 健壮 的 组 件 库 构成 ， 其 中 的 编程 模型 与 HTML 和 JavaScript 的 编程 模型 类 似 。 
你 需要 使 用 MXML 语 法 来 声明 布局 管理 和 工具 控件 ， 并 且 需 要 使 用 动态 脚本 机 制 来 添加 事件 处 
理 和 服务 调用 代码 ， 它 们 会 将 用 户 界面 链接 到 Java 类 、 数 据 模型 和 Web Services 等 上 。Flex 编 译 
器 接受 MXML 和 脚本 文件 ， 然 后 将 它们 编译 成 字 节 码 。 客 户 端 的 Flash 虚 拟 机 的 操作 方式 与 Java 
虚拟 机 类 似 ， 因 为 它 也 会 解释 编译 过 的 字 节 码 。Flash 字 节 码 的 格式 被 称 为 SWF，SWF 文 件 是 由 
Flex 编 译 器 产生 的 。 

注意 ， 在 http://openlaszlo.org 上 有 一 个 开源 的 Flex 的 可 替换 选择 ， 它 拥有 与 Flex 相 似 的 结构 ， 
对 于 某 些 应 用 ， EIR SR Ee: 还 有 一 些 其 他 的 工具 也 可 以 用 不 同 的 方式 来 创建 Flash 
应 用 。 
22.13.1 Hello, Flex 

请 观察 下 面 的 MXML 代 码 ， 它 定义 了 一 个 用 户 界面 (注意 ， 第 一 行 和 最 后 一 行 并 未 出 现在 
你 下 载 的 本 书 的 代码 包 中 ): 


/1/:1 gui/flex/helloflexl.mxml 
<?xml version="1.0" encoding="utf-8"?> 
<mx: Application 
xmins:mx="http: //www, macromedia. com/2603/mxml" 
backgroundColor="#ffffff"> 
<mx:Label id="output" text="Hello, Flex!" /> 
</mx: Appl ication> 
Whim 


MXML 文 件 是 XML 文档 ， 因 此 它们 以 XML 版 本 /编码 指示 开始 的 。 最 外 边 的 MXML 元 素 是 
Application 元 素 ， 它 是 Flex 用 户 界面 最 顶层 的 可 视 化 和 逻辑 容器 。 你 可 以 声明 表示 可 视 化 控件 
的 标签 ， 例 如 上 面 的 在 Application 元 素 内 部 的 Label 元 素 。 控 制 总 是 被 置 于 某 个 容器 的 内 部 ， 而 
容器 在 众多 的 机 制 中 封装 了 布局 管理 器 ， 因 此 容器 将 管理 在 其 中 的 控件 的 布局 。 在 最 简单 的 情 
况 中 ， 例 如 上 面 的 示例 ，Application 将 起 到 容器 的 作用 。Application 默 认 的 布局 管理 器 仅仅 是 
将 控件 按照 它们 被 声明 的 顺序 ， 在 界面 上 垂直 向 下 地 排列 。 

ActionScript 是 ECMAScript 或 JavaScript 的 某 个 版 本 ， 它 看 上 去 与 Java 很 相似 ， 并 且 除 了 支持 
动态 脚本 机 制 之 外 ， 还 支持 类 和 强 类 型 。 通 过 向 本 例 中 添加 脚本 ， 我 们 可 以 引入 行为 。 在 下 面 
的 示例 中 ，MXML Seript 控 件 被 用 来 直接 将 ActionScript 放 置 到 MXML 文 件 中 : 


/1/:! gui/flex/helloflex2.mxml 
<?xml version="1.0" encoding="utf-8"?> 
<mx:Application 
xmlns:mx="http://www. macromedia. com/2003/mxm" 
backgroundColor="#ffffff"> 
<mx: Script> 
<![CDATAL 
function updateOutput() { 
output.text = “Hello! ”+ input.text; 
} 
]]> 
</mx: Ser ipt> 
<mx:TextInput id="input" width="200" 
change="updateOutput()” /> 
<mx:Label id="output” text="Hello!" /> 
</mx: Applicat ion> 
Mdm 


TextInput 控 件 接收 用 户 的 输入 ， 而 Label 显 示 所 键入 的 数据 。 注 意 ， 每 个 控件 的 id 属性 在 肢 
本 中 都 被 当 作 变量 名 ， 从 而 变 成 可 访问 的 了 ， 因 此 脚本 可 以 引用 MXML 标 签 的 实例 。 在 
TextInput 域 中 ， 你 可 以 看 到 change 属 性 连接 到 了 updateOutput0 函 数 上 ， 使 得 无 论 发 生 了 何 种 
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修改 ， 这 个 函数 都 会 被 调用 。 
22.13.2 编译 MXML 

启动 使 用 Flex 之 旅 的 最 简单 方式 就 是 使 用 免费 试用 版 ， 这 可 以 从 www.macromedia.com/ 
software/flex/trial 处 下 载 * 。Flex 这 个 产品 打包 了 大 量 的 版 本 ， 从 免费 试用 版 到 企业 服务 器 版 ， 并 
且 Macromedia 还 为 开发 Flex 应 用 程序 提供 了 额外 的 工具 。 确 切 的 打包 机 制 在 不 断 地 变化 ， 所 以 
请 检查 Macromedia 网 站 以 了 解 具体 信息 。 还 应 该 注意 的 是 ， 你 可 能 需要 修改 在 Flex 安 装 的 bin 目 
录 中 jvm.config 文 件 : 

为 了 将 MXML 文 件 编译 为 Flash 字 节 码 ， 你 有 两 个 选择 : 

1) 你 可 以 将 MXML 文 件 放 在 Java Web 应 用 程序 中 ， 与 JSP 和 HTML 同 处 一 个 WAR 文 件 中 ， 
然后 在 游览 器 请 求 MXML 文 档 的 URL 时 ， 在 运行 时 编译 所 请 求 的 .mxml 文 件 。 

2) 你 可 以 用 Flex 命 令 行 编译 器 mxmlc 编 译 MXML 文 件 。 

第 一 个 选择 ， 即 基于 Web 的 运行 时 编译 ， 除 Flex 之 外 ， 还 需要 一 个 Servlet 容 器 (例如 Apache 
Tomcat) 。Servlet 容 器 的 WAR 文 件 必须 用 Flex 配 置信 息 进行 更 新 ， 例 如 添加 到 web.xml 描 述 符 中 
的 Servlet 映 射 ， 并 且 它 还 必须 包括 Flex 的 JAR 文 件 一 当 你 安装 Flex 时 ， 这 些 步 又 会 自动 得 到 处 
理 。 在 WAR 文 件 配置 好 之 后 ， 你 就 可 以 将 MXML 文 件 放 到 Web 应 用 程序 中 ,并 且 通 过 任何 浏览 
器 来 请 求 这 些 文档 的 URL。Flex 将 在 第 一 次 被 请 求 时 编译 该 应 用 程序 ， 这 与 JSP 模 型 类 似 ， 其 后 
将 在 HTML 外 壳 中 传递 编译 过 且 缓存 的 SWF。 

第 二 种 选择 不 需要 服务 器 。 当 你 在 命令 行 中 调用 Flex 的 mxmjc 编 译 器 时 ,就 会 产生 SWF 文件 ， 
可 以 按照 你 的 意愿 部 属 它们 。mxmle 可 执行 程序 位 于 Flex 安 装 的 bin 目 录 下 ， 调 用 它 时 不 提供 任 
何 参数 可 以 将 有 效 的 命令 行 选 项 列 出 来 。 通 常 ， 你 需要 指定 Flex 客 户 端 组 件 库 的 位 置 ， 来 作 
为 -flexlib 命 令 行 选项 ， 但 是 在 像 前 面 看 到 的 两 个 非常 简单 的 示例 中 ，Flex 编 译 器 将 假设 组 件 库 
的 位 置 。 因 此 可 以 像 下 面 这 样 编译 前 面 的 两 个 示例 : 


mxmlc.exe helloflex1.mxml 
mxmlc .exe helloflex2.mxml 


这 将 产生 一 个 helloflex2.swf 文 件 ， 它 可 以 在 Flash 中 运行 ， 或 者 与 HTML 一 起 置 于 任何 HTTP 
服务 器 之 上 (一旦 Flash 被 加 载 到 Web 浏 览 器 中 ， 你 通常 只 需 在 SWF 文件 上 双击 就 可 以 在 浏览 器 
中 启动 它 ) 。 

对 于 helloflex2.swf， 你 可 以 看 到 下 面 这 个 运行 在 Flash Player 中 的 用 户 界面 ， 


This was not too hard t 


Hello! This was not too hard to do... 

在 更 复杂 的 应 用 程序 中 ， 你 可 以 通过 引用 在 外 部 ActionScript 文 件 中 的 函数 ， 来 将 MXML 和 
ActionScript 分 离开 。 在 MXML 中 ， 可 以 使 用 下 面 用 于 Script 控件 的 语法 : 

<mx:Script source="MyExternalScript.as” /> 

这 行 代码 使 得 MXML 控 件 可 以 引用 位 于 名 为 MyExternalScript.as 的 文件 中 的 函数 ， 就 好 像 这 
些 函 数位 于 MXML 文 件 中 一 样 。 
22.13.3 MXML 与 ActionScript 

MXML 是 ActionScript 类 的 声明 式 快 捷 方式 。 无 论 你 看 到 何 种 MXML 标 签 ， 都 有 一 个 同名 的 
ActionScript 类 与 之 对 应 。 当 Flex 编 译 器 解析 MXML 时 ， 它 首先 将 XML 转换 为 ActionScript， 并 加 


O 注意 ， 你 必须 下 载 Flex， 而 不 是 FlexBuilder。 后 者 是 IDE 设 计 工具 。 
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载 所 引用 的 ActionScript 类 ， 然 后 将 这 些 ActionScript 编 译 链接 为 SWF。 

”你 可 以 只 用 ActionScript 而 不 使 用 任何 MXML 来 编写 完整 的 Flex 应 用 程序 。 因 此 ，MXML 只 
是 一 种 便利 工具 。 诸 如 容器 和 控件 这 样 的 用 户 界面 组 件 通常 都 是 用 MXML 声 明 的 ， 而 像 事件 处 
理 和 其 他 客户 端 逻辑 这 样 的 逻辑 则 是 通过 ActionScript 和 Java 处 理 的 。 

你 可 以 创建 自己 的 MXML 控 件 ， 并 通过 编写 ActionScript 类 来 用 MXML 引 用 它们 。 你 还 可 以 
将 现 有 的 MXML 容 器 和 控件 组 合 到 一 个 新 的 MXML 文 档 中 ， 这 样 它 就 可 以 在 其 他 的 MXML 文 档 
中 作为 一 个 标签 来 引用 了 。Macromedia 的 Web 网 站 提供 了 具体 实现 这 一 功能 的 信息 。 
22.13.4 容器 与 控制 

Flex 组 件 库 的 可 视 化 核心 是 一 个 容器 集合 ， 这 些 容器 管理 着 布局 ， 以 及 在 容器 中 的 控件 数 
组 。 容 器 包括 面板 、 生 直 和 水 平 箱 体 、 瓦 片 、 折 番 夹 、 分 格 箱 体 以 及 网 格 等， 控件 是 用 户 界面 
上 的 小 部 件 ， 例 如 按钮 、 文 本 区 域 、 滑 块 、 日 历 和 数据 网 格 等 等 。 

本 节 剩 余部 分 将 展示 一 个 可 以 显示 和 排序 音频 文件 列表 的 Flex 应 用 程序 。 这 个 应 用 程序 演 
示 了 容器 、 控 件 ， 以 及 如 何 从 Flash 连 接 到 Java。 

现在 我 们 开始 编写 MXML 文 档 ， 将 一 个 DataGrid 控 件 (更 加 复杂 的 Flex 控 件 之 一 ) 放置 到 
一 个 Panel 容 器 中 : 


/1:! gui/flex/songs.mxml 
<?xml version="1.0" encoding="utf-8"?> 
<mx:Application 
xmins:mx="http://www.macromed a.com/2@03/mxm1" 
backgroundColor="#B9CAD2" pageTitle="Flex Song Manager” 
initialize="getSongs()"> 
<mx: Script source="songScript.as" /> 
<mx:Style source="songStyles.css*/> 
<mx:Panel id="songl istPanel” 
titleStyleDeclaration="headerText” 
title="Flex MP3 Library”> 
<mx:HBox verticalAlign="bottom"> 
<mx:DataGrid id="songGrid” 
cellPress="selectSong(event)” rowCount="8"> 
<mx:columns> 
«mx: Array> 
<mx:DataGridColumn columnName="name" 
headerText="Song Name" width="120" /> 
s<mx:DataGridColumn columnName="artist” 
headerText="Artist” width="180" /> 
<mx:DataGridColumn columnName=" album" 
headerText="Album" width="160" /> 
</mx:Array> 
</mx:columns> 
</mx:DataGrid> 
<mx : VBox> 
<mx:HBox height="100" > 
<mx: Image id="albumImage” source="" 
height="80" width="100" 
mouseOverEffect="resizeBig” 
mouseOutEffect="resizeSmall" /> 
<mx:TextArea id="songinfo" 
styleName="boldText" height="100%" width="120" 
vScrollPolicy="off" borderStyle="none" /> 
</mx : HBOx> 
<mx:MediaPlayback id=*songPlayer* 
contentPath="" 
mediaTy; P3” 
height= 
width="230" ze 
controllerPolicy="on" 
autoPlay="false" 
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visible="false" /> 
</mx:VBox> 
</mx:HBox> 
<mx:ControlBar horizontalAlign="right"> 
<mx:Button id="refreshSongsButton" 
label="Refresh Songs" width="100" 
toolTip="Refresh Song List” 
click="songService.getSongs()" /> 
</mx: ControlBar> 
</mx:Panel> 
<mx: Ef fect> 
<mx Resize name="resizeBig” heightTo="100" 
duration="5i 1> 
<mx:Resize name="resizeSmall” heightTo="80" 
duration="500"/> 
</mx:Effect> 
<mx:RemoteObject id="songService” 
Source="gui .flLex.SongService” 
resutt="onSongs(event.resutt)" 
fault="alert(event.fault.faultstring, ‘Error')"> 
<mx:method name="getSongs"/> 





</mx: RemoteObject> 
</mx:Application> 
Ii~ 


DataGrid & HFHARAMRR RE. SPCR MEM LER — A REUSE TERM, 
应 该 知道 它 对 应 于 底层 ActionScript 中 的 某 个 属性 、 事 件 或 封装 的 对 象 。DataGrid 有 一 个 值 为 
songGrid 的 id 属性 ， 因 此 ActionScript 和 MXML 标 签 都 可 以 通过 将 songGrid 用 作 变 量 名 来 以 编程 
方式 引用 这 个 网 格 。DataGrid 所 包含 的 属性 比 这 里 所 展示 的 要 多 得 多 ， 完 整地 MXML 控 件 和 容 
器 的 API 可 以 在 http://livedocs.macromedia.com/flex/15/asdocs_en/index.html 处 找到 。 

DataGrid 后 面 是 一 个 包含 一 个 Image 的 VBox， 它 可 以 显示 专辑 的 封面 以 及 歌曲 信息 ， 还 包 
含 一 个 可 以 播放 MP3 文 件 的 MediaPlayback 控 件 。 这 个 示例 以 流 的 方式 来 播放 文件 的 内 容 ， 这 样 
可 以 减 小 编译 后 的 SWF 的 尺寸 。 当 你 在 Flex 应 用 程序 中 嵌入 图 片 、 音 频 和 视频 文件 而 不 是 以 流 
的 方式 来 打开 时 ， 这 些 文件 会 成 为 编译 后 生成 的 SWF 的 一 部 分 ， 并 跟随 你 的 用 户 界 面 内 容 一 起 
传递 ， 而 不 是 在 运行 时 以 流 的 方式 随 需 打 开 。 

Flash Player 包 含 内 幅 的 多 媒体 数字 信号 编 解码 器 ， 可 以 用 于 播放 和 以 流 的 方式 打开 各 种 格 
式 的 音频 和 视频 文件 。Flash 和 Flex 都 支持 使 用 Web 上 最 通用 的 图 片 格式 ， 并 且 Flex 还 具有 将 可 扩 
展 矢量 图 (SVG) 文件 转换 为 可 以 内 嵌 在 Flex 客 户 端 中 的 SWF 资源 的 能 力 。 

22.13.5 效果 与 样式 

Flash Player 使 用 向 量 来 呈现 图 形 ， 因 此 它 可 以 在 运行 时 执行 极 富 表现 力 的 转换 。Flex 效 
果 可 以 让 你 浅 尝 这 些 动画 类 型 。 效 果 是 指 可 以 通过 使 用 MXML 语 法 而 应 用 到 控件 和 容器 上 的 
转换 。 

在 MXML 中 展示 的 Effect 标签 会 产生 两 个 结果 : 第 一 个 代 套 标签 在 鼠标 滑 过 图 片 时 动态 地 扩 ， 
大 图 片 ， 而 第 二 个 则 是 在 鼠标 离开 图 片 时 动态 地 缩小 图 片 。 这 些 效果 被 应 用 于 albumImage 对 应 
的 Image 控 件 上 的 鼠标 事件 上 。 

了 lex 还 为 通用 动画 提供 了 效果 ， 例 如 渐变 、 氛 去 和 调整 aljpha 通 道 。 除 了 内 建 的 效果 ，Flex 还 
支持 用 于 创新 动画 效果 的 Flash 绘 制 API。 对 这 个 主题 更 深入 的 研究 将 涉及 图 形 设计 和 动画 ， 而 
这 超出 了 本 节 的 范围 。 

通过 Flex 对 层 登 样式 表 (Cascading Style Sheets, CSS) 的 支持 ， 我 们 还 可 以 进行 样式 标准 
化 操作 。 如 果 你 将 一 个 CSS 文 件 附着 到 MXML 文 件 上 , 那么 Flex 控 件 将 都 遵循 其 中 的 样式 。 例 如 ， 
songStyles.css 包 含 下 面 的 CSS 声 明 : 
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1/:! gui/flex/songStyles.css 
sheaderText { 
font-family: Arial, “_sans"; 
font-size: 16; 
font-weight: bold: 
} 


-botdText { 
font-family: Arial, "_sans™; 
font-size: 11; 
font-weight: bold: 


} 
H~ 


这 个 文件 通过 MXML 文 件 中 的 Style 标 签 被 导入 到 了 歌曲 类 库 应 用 程序 中 ， 并 在 其 中 得 到 了 
使 用 。 在 样式 表 被 导入 之 后 ， 它 的 声明 就 可 以 应 用 于 MXML 文 件 中 的 Flex 控 件 了 。 作 为 示例 ，id 
为 songInfo 的 TextArea 控 件 使 用 了 样式 表 的 boldText 声 明 。 

22.13.6 事件 

用 户 界面 是 一 种 状态 机 ， 它 在 状态 发 生变 化 时 执行 动作 。 在 Flex 中 ， 这 些 变化 是 通过 事件 
来 管理 的 。Flex 类 库 包 含 大 量 各 种 不 同 的 控件 ， 它 们 具有 大 量 的 事件 ， 和 覆盖 了 鼠标 移动 和 键盘 
点 击 的 所 有 方面 。 

例如 ，Button 的 click 属 性 表示 在 按钮 控件 上 可 用 的 事件 。 赋 给 click 的 值 可 以 是 一 个 函数 或 
内 联 的 少量 脚本 。 例 如 ， 在 MXML 文 件 中 ，ControlBar 持 有 可 以 刷新 歌曲 列表 的 refreshSongs- 
Button。 从 标签 就 可 以 看 出 ， 当 click 事件 发 生 时 ，songService.getSongs0 将 被 调用 。 在 本 例 中 ， 
Button 的 click 事件 引用 了 RemoteObject， 它 与 Java 方 法 相对 应 。 

22.13.7 连接 到 Java 

在 MXML 文 件 末尾 的 RemoteObject 标 签 设置 了 到 外 部 Java 类 gui.flex.SongService 的 连接 。 
Flex 客 户 端 将 使 用 这 个 Java 类 中 的 getSongs0 方 法 来 为 DataGrid 获 取 数 据 。 为 了 实现 这 一 点 ， 它 
必须 看 起 来 像 是 一 个 服务 - 个 用 户 可 以 用 来 交换 消息 的 端点 。 在 RemoteObject 标 签 中 定义 
的 服务 有 一 个 source 属 性 ， 它 指示 的 是 RemoteObject 的 Java 类 ， 并 且 还 指定 了 一 个 在 Java 方 法 返 
回 时 被 调用 的 ActionScript 回 调 函 数 onSongs0。 供 套 的 method 标 签 声 明了 getSongs() 方 法 ， 它 可 
以 使 Java 方 法 对 Flex 应 用 程序 中 的 其 他 部 分 都 是 可 访问 的 。 

在 Flex 中 所 有 的 服务 调用 ， 都 是 通过 事件 调用 这 些 回调 函数 而 异步 返回 的 。RemoteObject 
在 错误 事件 中 还 会 产生 一 个 警告 对 话 框 构件 。 

现在 可 以 使 用 ActionScript 从 Flash 中 调用 getSongs0 方 法 了 : 

songService.getSongs(); 
根据 MXML 的 配置 ， 这 将 调用 SongService 类 中 的 getSongs0 方 法 : 

//: gui/flex/SongService. java 


package gui. flex; 
import java.util.*; 





public class SongService { 
private List<Song> songs = new ArrayList<Song>(): 
public SongService() { fillTestData(); } 
public List<Song> getSongs() { return songs; } 
public void addSong(Song song) { songs.add(song); } 
public void removeSong(Song song) { songs.remove(song); } 
private void fillTestData() { 
addSong(new Song("Chocolate", "Snow Patrol", 
“Final Straw", "sp-final-straw. jpg”. 
“chocolate.mp3")); 
addSong(new Song("Concerto No. 2 in E", “Hilary Hahn", 
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"Bach: Violin Concertos", “hahn. jpg”. 
“bachviolin2.mp3")) ; 

addSong(new Song("*Round Midnight” 
"The Artistry of Wes Montgomery’ 
"wesmontgomery. jpg", “roundmidnight.mp3")); 


“Wes Montgomery", 





} 
YM 


每 个 Song 对 象 都 只 是 一 个 数据 容器 : 


/1/: gui/flex/Song. java 
package gui. flex; 


public class Song implements java.io.Serializable { 

private String nane; 

private String artist 

private String albu 

private String albumImageUrl; 

private String songMediaUrl: 

public Song() {} 5 

public Song(String name, String artist, String album, 

String albumImageUrl, ‘String songMediaUrl) { 
this.name = name; 
this.artist = artist; 
this.album = album; 
this.albumImageUrl = albumImageUrt; 
this.songMediaUrl = songMediaUrl; 





} 
public void setAlbum(String album) { this.atbum = album;} 
public String getAlbum() { return album; } 
public void setAlbumImageUrt (String albumImageUrl) { 
this.albumImageUrl = albumImageUrl; ta 
) 
public String getAlbumImageUrl() { return albumImageUrl;} 
public void setArtist(String artist) { 
this.artist = artist; 


} 

public String getArtist() { return artist; } 

public void setName(String name) { this.name = name; } 
public String getName() { return name; } 

public void setSongMediaUrl (String songMediaUrl) { 





1425] this.songMediaUrl = songMediaUrl; 
} 
; public String getSongMediaUrl() { return songMediaUrl; } 
H~ 


当 应 用 程序 被 初始 化 ， 或 者 按 下 refreshSongsButton 时 ，getSongs0 都 将 被 调用 ， 并 且 在 返回 后 ， 
ActionScript 函 数 onSongs(event,result) 将 被 调用 以 组 装 songGrid。 
下 面 是 ActionScript 的 列表 ， 在 MXML 文 件 中 用 Script 控 件 将 其 导入 : 


//: gui/flex/songScript.as 
function getSongs() { 

songService.getSongs() ; 
$ 


function selectSong(event) { 
var song = songGrid.getItemAt (event. itemIndex) ; 
showSongInfo(song) ; 

} 


function showSongInfo(song) { 
songInfo.text = song.name + newline; 
songinfo.text += song.artist + newlin 
songInfo.text += song.album + newline 
albumImage.source = song.albumImageUrl; 
songPlayer.contentPath = song. songMediaUrl; 
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songPlayer.visible = true: 


function onSongs(songs) { 
songGrid.dataProvider = songs: 
yh 


为 了 处 理 对 DataGrid 单 元 格 的 选中 操作 ， 我 们 在 MXML 文 件 的 DataGrid 声 明 中 添加 了 cellPress 
事件 属性 ; 

cellPress="selectSong(event)” 

当 用 户 在 DataGrid 中 点 击 一 首 歌 时 ， 将 会 调用 上 面 的 ActionScript 中 的 selectSong0。 
22.13.8 数据 模型 与 数据 绑 定 

控件 可 以 直接 调用 服务 ，ActionScript 事 件 回调 使 你 在 服务 返回 数据 时 ， 能 够 以 编程 方式 更 
新 可 视 化 控件 。 尽 管 更 新 控件 的 脚本 很 直观 ， 但 是 它 可 能 会 变 得 宛 长 而 麻烦 ， 又 因为 其 功能 很 
通用 ， 所 以 Flex 用 数据 绑 定 机 制 自动 地 处 理 这 种 行为 。 

在 最 简单 的 数据 绑 定形 式 中 ， 控 件 可 以 直接 引用 数据 而 不 需要 用 粘 合 代码 把 数据 复制 到 控 
件 中 。 当 数据 更 新 时 ， 引 用 它 的 控件 也 会 自动 更 新 ， 而 不 需要 任何 程序 员 的 干预 。Flex 基 础 设 
施 会 恰当 地 响应 数据 变化 事件 ， 并 且 更 新 所 有 绑 定 到 该 数据 的 控件 。 

下 面 是 数据 绑 定 语法 的 简单 示例 : 


<mx:Slider id="mySlider"/> 
<mx:Text text="{mySlider.value}"/> 


为 了 执行 数据 绑 定 ， 你 需要 将 引用 置 于 花 括号 1 中 。 在 花 括号 中 的 所 有 事物 都 被 认为 是 由 Flex 计 
算 的 表达 式 。 

， 第 一 个 控件 Slider 部 件 的 值 由 第 二 个 控件 Text 域 显示 。 当 Slider 变 化 时 ，Text 域 的 text 属 性 会 
被 自动 更 新 。 通 过 这 种 方式 ， 开 发 者 不 需要 为 了 更 新 Text 域 而 处 理 Slider 变 化 事件 。 

某 些 控件 ， 例 如 Tree 控件 和 歌曲 库 应 用 程序 中 的 DataGrid 控 件 ， 会 更 加 复杂 。 这 些 控件 有 
一 个 dataprovider 属 性 ， 可 以 使 绑 定 到 数据 集 更 加 容易 。ActionScript 的 onSongs0 〇 函数 展示 了 如 
何 将 SongService.getSongs0 方 法 绑 定 到 Flex 的 DataGrid 的 dataprovider 上 。 正 如 在 MXML 文 件 的 
RemoteObject 标 签 中 所 声明 的 ， 这 个 函数 是 在 Java 方 法 返回 时 ActionScript 调 用 的 回调 。 

更 复杂 的 应 用 程序 需要 更 复杂 的 数据 建 模 ， 例 如 使 用 数据 传输 对 象 的 企业 应 用 系统 ， 或 者 
数据 遵循 复杂 模式 的 基于 消息 机 制 的 应 用 程序 ， 都 会 鼓励 我 们 进一步 将 数据 源 与 控件 解 罩 。 
在 Flex 开 发 中 ， 我 们 通过 声明 “模型 ”对 象 来 执行 这 种 解 辜 ， 而 这 种 对 象 是 用 于 数据 的 通用 
MXML 容 器 中 的 。 模 型 不 包含 任何 肥 辑 ， 它 是 在 企业 应 用 程序 开发 中 的 数据 传输 对 象 ， 或 者 
共 他 编程 语言 的 类 似 结构 的 镜像 。 通 过 使 用 模型 ， 我 们 可 以 将 控件 数据 绑 定 到 模型 上 ， 同 时 
可 以 让 模型 将 它 的 属性 绑 定 到 服务 的 输入 和 输出 上 。 这 可 以 将 数据 源 、 服 务 与 数据 的 可 视 化 
消费 者 解 耦 ， 从 而 促进 对 模型 -视图 一 控制 器 (MVC) 模式 的 使 用 。 在 更 大 更 复杂 的 应 用 程序 
中 ， 与 由 插入 模 型 而 带 来 的 复杂 性 与 清晰 解 耦 的 MVC 应 用 程序 的 价值 相 比 ， 这 绝对 是 花 小 钱 
办 大 事 。 

除了 Java 对 象 ，Flex 还 可 以 通过 使 用 WebService 和 HttpService 控 件 来 分 别 访问 基于 SOAP 的 
Web 服 务 和 更 容易 调用 的 HTTP 服 务 。 访 问 所 有 的 服务 都 会 受到 安全 授权 的 限制 。 

22.13.9 构建 和 部 署 

在 使 用 前 面 的 示例 时 ， 你 在 命令 行 中 可 以 不 提供 -flexlib 标 签 ， 但 是 为 了 编译 这 个 程序 ， 必 
须 使 用 -flexlib 标 签 指定 flex-config.xml 文 件 的 位 置 。 对 我 的 安装 来 说 ， 下 面 的 命令 是 可 以 工作 的 ， 
但 是 你 必须 将 其 修改 为 适应 你 自己 的 配置 (命令 是 单行 的 ， 即 中 间 被 包装 的 那 一 行 ) : 
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/1:1 gui/flex/build-command. txt 
mxmic -flexlib C:/"Program 

Files" /Macromedia/Flex/jrun4/servers/default/flex/WEB- 
INF/flex songs.mxml 

Mh 


这 条 命令 将 把 应 用 程序 构建 为 一 个 可 以 用 浏览 器 查看 的 SWF 文件 ， 但 是 本 书 的 代码 发 布 文 
件 中 没有 包含 任何 MP3 文 件 或 JPG 文 件 ， 因 此 当 你 运行 这 个 应 用 程序 时 ， 除 了 框架 之 外 不 会 看 到 
任何 东西 。 

另外 ， 你 必须 配置 服务 器 使 得 Flex 应 用 程序 可 以 成 功 地 与 Java 文 件 对 话 。Flex 试 用 包 中 包含 
一 个 JRun 服 务 器 ， 一 旦 你 安装 了 Flex， 就 可 以 通过 计算 机 菜单 或 者 命令 行 来 启动 它 ; 

jrun -start default 

你 可 以 通过 在 Web 浏 览 器 中 打开 http://localhost:8700/samples， 并 查看 各 种 示例 来 验证 这 个 服 
务 器 是 否 成 功 启动 了 (这 还 是 一 种 熟悉 Flex 能 力 的 好 方式 ) 。 

与 在 命令 行 编译 应 用 程序 不 同 ， 你 可 以 通过 服务 器 编译 它 。 要 实现 这 一 点 ， 需 要 将 歌曲 源 
文件 、CSS 样 式 表 等 放 到 jrun4/servers/default/flex 目 录 下 ， 并 通过 打开 http://localhost:8700/flex 
/songs.mxml 来 在 浏览 器 中 访问 它们 。 

要 想 成 功 运行 这 个 Web 应 用 ， 你 必须 同时 配置 Java 端 和 Flex 端 。 

Java; 编译 后 的 Song.java 和 SongService.java 文 件 必须 置 于 WEB-INF/classes 目 录 中 ， 这 正 
是 按照 J2EE 规 范 放 置 WAR 类 的 地 方 。 或 者 ， 你 可 以 将 这 些 文件 建 档 ， 然 后 将 产生 的 JAR 文 件 放 
到 WEB-INF/lib 目 录 下 。 这 些 类 必须 位 于 匹配 其 Java 包 结构 的 目录 中 ， 如 果 使 用 JRun， 那 么 它们 
就 应 该 位 于 jrun4/servers/default/flex/WEB-INF/classes/gui/nlex/Song.class 和 jrun4/servers/ 
defaultflex/WEB-INF/classes/gui/nex/SongService.class。 你 还 需要 使 图 片 和 MP3 支 持 文件 在 Web 
应 用 中 可 用 (对 于 JRun 来 说 ，jrun4/serversdefaultUnlex 是 Web 应 用 的 根 ) 。 

Flex; 为 了 安全 性 ， 除 非 你 通过 修改 flex-config.xml 文 件 来 赋予 权限 ， 否 则 Flex 将 不 能 访问 
Java 对 象 。 对 于 JRun， 这 个 文件 位 于 jrun4/servers/default/flex/WEB-INF/flex/flex-config.xml。 
转 到 这 个 文件 的 <remote-object> 项 ， 查 看 其 中 的 <whitelist> 一 节 ， 你 会 看 到 下 面 的 提示 


For security, the whitelist is locked down by default. Uncomment the 
source element below to enable access to.all classes during development. 


We strongly recommend not allowing access to all source files in 
production, since this exposes Java and Flex system classes. 
<source>*</source> 

~> 


为 了 允许 访问 而 去 掉 <souree> 项 的 注释 ， 将 使 得 <source>*</sourece> 可 以 被 读 取 。 这 个 项 和 
其 他 项 的 含义 在 Flex 配 置 文档 中 都 有 所 描述 。 

练习 38: (3) 构建 上 面 所 示 的 “数据 绑 定 语法 的 简单 示例 ”。 

练习 39，(4) 本 书 提供 的 下 载 代码 不 包含 SongService.java 中 所 示 的 MP3 和 JPG 文 件 。 找 一 些 
MP3 和 JPG 文 件 ， 修 改 SongService.java， 使 其 包含 这 些 文件 的 名 字 ， 下 载 Flex 使 用 版 并 构建 这 个 
应 用 程序 。 


22.14 创建 SWT 应 用 


正如 前 面 提 到 的 ，Swing 采 用 的 方式 是 将 所 有 的 UI 组 件 逐 个 像素 地 构建 ， 以 便 提供 所 有 想 要 
的 组 件 ， 无 论 底层 操作 系统 是 否 拥 有 这 些 组 件 。SWT 采 用 了 中 间 路 线 ， 如 果 操 作 系统 提供 本 地 组 
件 ， 那 么 就 使 用 这 些 本 地 组 件 ， 如 果 不 提 供 就 合成 这 些 组 件 。 其 结果 就 是 这 种 应 用 程序 对 用 户 而 
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言 ， 感 觉 就 像 是 本 地 应 用 程序 一 样 ， 并 且 与 等 价 的 Swing 程序 相 比 ， 在 性 能 上 有 明显 的 提高 。 另 
外 ，SWT 与 Swing 相 比 ， 其 编 成 模型 要 更 简单 一 些 ， 这 是 相当 多 的 应 用 程序 都 希望 具有 的 特性 。 

因为 SWT 尽 可 能 多 地 使 用 本 地 操作 系统 来 完成 工作 ， 因 此 它 可 以 自动 地 利用 Swing 可 能 无 法 
利用 的 操作 系统 特性 ， 例 如 ，Windows 具 有 “ 子 像素 呈现 ”机 制 ， 它 可 以 使 字体 在 LCD 屏 幕 上 
更 清楚 地 显示 。 甚 至 还 可 以 用 SWT 创 建 applet。 

本 节 并 不 打算 对 SWT 做 全 面 介绍 ， 只 是 希望 能 够 让 你 喜欢 上 它 ， 并 且 看 到 SWT 与 Swing 的 
对 比 。 你 会 发 现 SWT 有 很 多 部 件 ， 它 们 都 简单 易 用 。 你 可 以 在 www.eclipse.org 网 站 提供 的 完整 
文档 和 许多 示例 中 探究 SWT 的 细节 。 还 有 大 量 关 于 用 SWT 编 程 的 书籍 ， 并 且 这 方面 的 书 还 在 层 
出 不 穷 。 

22.14.1 安装 SWT 

SWT 应 用 程序 要 求 从 Eclipse 项 目 中 下 载 并 安装 SWT 类 库 。 访 问 www.eclipse.org/downloads/ 并 
选择 一 个 镜像 。 点 击 能 链接 到 当前 Eclipse 构 建 的 链接 ， 并 用 以 swt 开 头 、 包 含 你 的 平台 名 字 ( 例 
如 win32) 在 内 的 名 字 来 定位 压缩 文件 。 在 这 个 文件 的 内 部 能 够 找到 swt.jar， 最 简单 的 安装 
swt.jar 的 方式 就 是 将 其 放置 到 你 的 jre/lib/ext 目 录 中 (用 这 种 方式 你 不 必 对 类 路 径 作 任何 修改 )。 
当 你 解压 缩 SWT 类 库 时 ， 需 要 找到 所 需 的 额外 文件 ， 把 它们 安装 到 你 的 平台 中 恰当 的 位 置 上 。 
例如 ，Win32 发 布 中 就 包含 DLL 文件 ， 它 们 需要 被 置 于 java.library.path (这 通常 与 PATH 环境 变 
量 相同 ， 但 是 你 可 以 通过 运行 objectIShowProperties.java 来 发 现 java.library.path 的 实际 值 ) 中 
的 某 处 。 一 旦 执行 完 这 些 动作 ， 就 应 该 能 够 透明 地 编译 和 执行 SWT 应 用 程序 了 ， 就 好 像 它们 无 
异 于 其 他 任何 Java 程 序 一 样 。SWT 的 文档 在 另外 一 个 单独 的 下 载 中 。 

另 一 种 可 选 方式 是 只 安装 Eclipse 编 辑 器 ， 它 包含 SWT 和 你 可 以 通过 Eclipse 帮 助 系 统 去 浏览 
的 SWT 文 档 。 

22.14.2 Hello, SWT 
让 我 们 以 最 简单 的 “hello world” 风 格 的 应 用 程序 开始 : 


/1: swt/HelloSWT. java 
// {Requires: org.eclipse.swt.widgets.Display: You must 
// install the SWT library from http://www.eclipse.org } 
import org.eclipse.swt.widgets.*; 


public class HelloSWT { 
public static void main(String [] args) { 
Display display = new Display(); 
Shell shell = new Shell (display); 
shell.setText ("Hi there, SWT!"); // Title bar 
shell.open(); 
while(!shell. isDisposed()) 
if (!display.readAndDispatch()) 
display. sleep(); 
display. dispose() ; 


} 
dM hm 
如 果 下 载 了 本 书 的 源 代码 ， 你 就 会 发 现 Requires 注 释 最 终 使 得 Ant 的 build.xml 成 为 了 构建 swt 
子 目 录 的 前 提 条 件 ， 所 有 导入 org.eclipse.swt 的 文件 都 需要 你 从 www.eclipse.org 来 安装 SWT 类 库 。 
Display 管 理 SWT 和 底层 操作 系统 之 间 的 连接 ， 它 是 操作 系统 和 SWT 之 间 的 桥 的 一 部 分 。 
Shell 是 顶层 主 窗口 ， 所 有 其 他 组 件 都 构建 于 其 中 ， 当 你 调用 setText0 时 ， 参 数 会 变 为 窗口 标题 
栏 上 的 标签 。 


© Chris Grindstaff 对 转译 SWT 示 例 和 提供 SWT 方 面 贡献 良 多 。 





[ag 








1431 











846 #n2# 





为 了 显示 窗口 以 及 这 样 的 应 用 程序 ， 你 必须 在 Shell 上 调用 open0。 

尽管 Swing 对 你 隐藏 了 事件 处 理 循 环 ， 但 是 SWT 会 强制 你 显 式 地 编写 它 。 在 循环 的 顶部 ， 检 
查 shell 是 否 已 经 被 释放 一 一 注意 ， 这 给 了 你 一 个 插入 代码 去 执行 清理 动作 的 选择 ， 但 是 这 意味 着 
main0 线 程 将 会 是 用 户 界面 线程 。 在 Swing 中 ， 后 台 会 创建 第 二 个 事件 分 发 线程 ,但 是 在 SWT 中 ， 
你 的 main0 线 程 将 处 理 UI。 由 于 默认 情况 下 只 有 一 个 线程 而 不 是 两 个 ， 这 使 得 在 某 种 程度 
上 ， 你 不 太 可 能 在 用 线程 来 处 理 UI 时 搞 得 一 塌 糊 涂 。 

注意 ， 你 不 必 像 使 用 Swing 那样 担心 向 用 户 界面 线程 提交 任务 ，SWT 不 仅 会 替 你 仔细 关照 这 
一 点 ， 而 且 如 果 你 试图 在 错误 的 线程 中 操作 部 件 ， 那 它 还 会 抛 出 异常 。 但 是 ， 如 果 你 需要 产生 
其 他 线程 去 执行 长 期 运行 的 操作 ， 那 么 你 仍旧 需要 按照 使 用 Swing 时 的 方式 去 提交 变化 。 为 了 这 
一 点 ，SWT 提 供 了 三 个 可 以 在 Display 对 象 上 调用 的 方法 : asyncExec(Runnable)、syncExec 
(Runnable) fitimerExec(int,Runnable), 

在 此 处 ， 你 的 main0) 线 程 的 活动 是 在 Display 对 象 上 调用 readAndDispatchO (这 意味 每 个 应 
用 程序 只 有 一 个 Display 对 象 ) 。 如 果 在 事件 队列 中 存在 更 多 的 事件 在 等 待 处 理 ， 那 么 
readAndDispatch() 方 法 将 返回 true。 在 这 种 情况 下 ， 你 希望 立即 再 次 调用 它 。 然 而 ， 如 果 没 有 
任何 事件 被 悬挂 ， 那 么 你 可 以 调用 Display 对 象 的 sleep( 方 法 ， 以 便 在 再 次 检查 事件 队列 之 前 等 

待 一 小 段 时 间 。 

一 旦 程序 结束 ， 你 必须 显 式 地 调用 Display 对 象 上 的 dispose0 方 法 。SWT 经 常 要 求 你 显 式 地 
释放 资源 ， 因 为 这 些 通常 都 是 来 自 底层 操作 系统 的 资源 ， 如 果 不 释放 可 能 会 被 耗 尽 。 

为 了 证 明 Shell 是 主 窗口 ， 下 面 的 程序 将 创建 大 量 的 Shell 对 象 : 


//: sWt/ShellsAreMainWindows. java 
import org.eclipse.swt.widgets.*; 





public class ShellsAreMainWindows { 
static Shell[] shells = new Shel1[10]; 
public static void main(String [] args) { 
Display display = new Display(); 
for(int 1 = 9; 1 < shells.length; i++) { 
shells[i] = new Shell (display): 
shells[i].setText("Shell #" + i); 
shells [1] .open(); 


while(!shellsDisposed()) 
if(!display.readAndDispatch()) 
display.sleep(); 
display. dispose(); 


} 

Static boolean shellsDisposed() { 
for(int i = Ø; 1 < shells.length; i++) 

if(shel1s [i]. isDisposed()) 
return true; 

return false; 

} 

} Mi~ 


当 你 运行 它 时 ， 将 获得 10 个 主 窗口 。 在 以 这 种 方式 编写 的 程序 中 ， 如 果 关 闭 任何 一 个 窗口 ， 那 
i 么 所 有 的 窗口 都 会 被 关闭 。 
SWT 也 使 用 了 布局 管理 器 ， 虽 然 它 与 Swing 使 用 的 不 同 ， 但 是 思想 一 致 。 下 面 是 稍微 复杂 一 
些 的 示例 ， 它 接收 从 System.getProperties0 中 获得 的 文本 ， 并 将 其 添加 到 shell 中 : 


//: swt/DisplayProperties.java 
import org.eclipse.swt.*; 

import org.eclipse.swt.widgets.*; 
1433] import org.eclipse.swt.layout.*: 
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import java.io.*; 


public class DisplayProperties { 
public static void main(String [} args) { 
Display display = new Display(): 
Shell shell = new Shell (display) : 
shell. setText ("Display Properties"); 
shell. setLayout (new FillLayout()); 
Text text = new Text(shell, SWT.WRAP | SWT.V_SCROLL) ; 
StringWriter props = new StringWriter(); 
System. getProperties().list(new PrintWriter (props)); 
text.setText (props. toString()); 
shell.open(); 
while(!shell. isDisposed()) 
if (Idisplay.readAndDispatch()) 
display.sleep(): 
display.dispose() ; 


} 

} Mi~ 

在 SWT 中 ， 所 有 部 件 必须 都 有 一 个 具有 泛 化 类 型 Conposite 的 父 对 象 ， 并 且 必须 在 部 件 构造 
器 中 将 这 个 父 对象 作 为 第 一 个 参数 来 提供 。 在 Text 构 造 器 中 你 就 可 以 看 到 这 一 点 ， 其 中 shell 是 
第 一 个 参数 。 实 际 上 ， 所 有 构造 器 还 都 要 接受 一 个 标志 参数 ， 它 使 得 你 可 以 根据 特定 的 部 件 所 
能 接受 的 情况 ， 提 供 任意 数量 的 样式 指示 信息 。 多 个 样式 指示 信息 是 按 位 或 在 一 起 的 ， 就 像 在 
本 例 中 看 到 的 那样 。 

在 设置 Text0 对 象 时 ， 我 添加 了 样式 标志 ， 以 使 得 它 将 文本 包装 起 来 ， 并 且 如 果 需 要 的 话 ， 
它 会 自动 地 添加 一 个 垂直 滚动 条 。 你 会 发 现 SWT 是 绝对 是 基于 构造 器 的 ， 各 种 部 件 都 有 大 量 的 
不 通过 构造 器 就 很 难 或 者 根本 不 可 能 修改 的 属性 ， 所 以 你 应 该 总 是 查看 部 件 构造 器 文档 ， 了 解 
可 接受 的 标志 。 注 意 ， 某 些 构造 器 即便 在 文档 中 没有 列 出 任何 “可 接受 ”的 标志 ， 它 们 也 要 求 
要 有 一 个 标志 参数 ， 这 样 就 可 以 使 得 未 来 在 做 扩充 时 ， 不 需要 修改 接口 。 
22.14.3 根除 元 余 代码 

在 继续 学 习 之 前 请 注意 ， 有 些 事情 是 你 在 每 个 SWT 应 用 程序 中 都 会 做 的 ， 这 与 Swing 程 序 中 
的 重复 动作 一 样 。 对 于 SWT， 你 总 是 得 创建 Display， 从 Display 中 创建 Shell， 然 后 创建 
readAndDispatch0 等 等 。 当 然 ， 对 于 某 些 特殊 情况 ， 你 可 以 不 做 这 些 ， 但 是 它 非常 普遍 ， 绝 对 
值得 去 根除 这 些 重复 代码 ， 就 像 在 net.mindview.util.SwingConsole 中 所 作 的 那样 。 

我 们 需要 强制 每 个 应 用 程序 遵循 下 面 的 接口 : 


/1: swt/util/SWTApplication. java 
package swt.util; 
import org.eclipse.swt.widgets.*: 


public interface SWTApplication { 
void createContents (Composite parent); 
} Mi~ 


应 用 程序 应 该 提交 给 了 Composite 对 象 (Shell 是 它 的 一 个 子 类 ) ， 并 且 应 该 用 它 在 
createContents0 内 部 创建 其 所 有 的 内 容 。 SWTConsole.run0 会 在 恰当 的 地 方 调用 createContents0， 
根据 用 户 传递 给 run0 的 参数 来 设置 shell 的 尺寸 ， 打 开 shell， 然 后 运行 事件 循环 ， 最 终 在 程序 退出 
时 释放 shell: 


/1: swt/util/SWTConsole. java 
package swt.util; 
‘import org.eclipse.swt.widgets.*; 


public class SWTConsole { 
public static void 
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run(SWTApplication swtApp, int width, int height) { 
Display display = new Display) ; 
Shell shell = new Shell (display); 
shell.setText(swtApp.getClass().getSimpleName()) ; 
swtApp.createContents (shell. 
shell.setSize(width, height 
shell .open() ; 
while(!shell.isDisposed()) { 

if(!display.readAndDispatch()) 
display.sleep() 六 





display.dispose(); 
} M~ 
这 个 类 还 会 将 标题 栏 设置 为 SWTApplication 类 的 名 字 ， 并 设置 Shell 的 width 和 height。 
我 们 可 以 创建 一 个 DisplayPropertiesjava 的 变 体 ， 它 可 以 用 SWTConsole 来 显示 机 器 环境 : 


//: swt/DisplayEnvironment. java 
import swt.util.*; 
import org.eclipse. swt. 
import org.eclipse.swt.widgets. 
import org.eclipse.swt. layout. 
import java.util,*; 








public class DisplayEnvironment implements SWTApplication { 
public void createContents (Composite parent) { 
parent.setLayout(new FillLayout()); 
Text text = new Text(parent, SWT.WRAP | SWT.V_SCROLL); 
for (Hap. Entry entry: System. getenv() entrySet()) { 
text. append(entry.getKey() + ": * 
entry.getValue() + "\n"); 





} 


} 
public static void main(String [] args) { 
SWTConsole.run(new DisplayEnvironment(), 800, 600); 


} 
} Mi~ 


SWTConsole 使 得 我 们 可 以 聚焦 于 应 用 程序 中 的 有 趣 方面 ， 而 不 是 重复 性 的 方面 。 

练习 40: (4) 修改 DisplayPropertiesjava， 让 其 使 用 5WTConsole。 

练习 41: (4) 修改 DisplayEnvironment.java， 让 其 不 使 用 SWTConsole。 
22.14.4 菜单 

为 了 演示 基本 的 菜单 ， 下 面 的 程序 将 读 入 它 自己 的 源 代码 ， 将 其 断 开 为 单词 ， 然 后 用 这 些 
单词 组 装 菜单: 


1/: swt/Menus.java 
// Fun with menus 
import swt.util 
import org.eclipse.swt.*; 
import org.eclipse. swt.widgets.*; 
import java.util.*; 
import net.mindview.util. 
public class Menus implements SWTApplication { 
private static Shell shell; 
public void createContents (Composite parent) { 
shell = parent. getShel1(); 
Menu bar = new Menu(shell, SWT.BAR); 
shell. setMenuBar (bar) ; 
Set<String> words = new TreeSet<String>( 
new TextFile("Menus.java", "\\W+")); 
Iterator<String> it = words. iterator(); 
while(it.next().matches("[8-3]+")) 
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; // Move past the numbers. 
Menultem[] mItem = new MenuItem[7] ; 
for(int i = 0; i < mItem. length; i++) { 
mItem[i] = new Menultem(bar, SWT.CASCADE); 
mItem[i] .setText (it.next()); 
Menu submenu = new Menu(shell, SWT.DROP_DOWN) ; 
mItem[i] .setMenu (submenu) ; 
} 
int i = 0; 
while(it.hasNext()) { 
addItem(bar, it, mItem[i]); 
i= (i + 1) % mItem. length; 
} 





} 

static Listener listener = new Listener() { 
public void handleEvent(Event e) { 

System.out.printin(e. toString()) ; 

} 

a 

void 

addItem(Menu bar, Iterator<String> it, Menultem mItem) { 
MenuItem item = new Menultem(mItem.getMenu(),SWT. PUSH) ; 
item. addListener(SWT.Selection, listener); 
item. setText (it.next()); 

) Z 

public static void main(String{] args) { 
SWTConsole.run(new Menus(), 660, 208); 


} 
} H~ 
Menu 必 须 置 于 某 个 Shell 之 上 ， 并 且 Composite 允 许 你 用 getShell0 获 取 它 的 shell。TextFile 来 
自 net.mindview.util， 并 且 在 前 面 已 经 描述 过 。 这 里 TreeSet 是 用 单词 填充 的 ， 因 此 它们 将 按照 排 
序 顺序 地 出 现 ， 基 中 最 初 的 元 素 是 数字 ， 它 们 将 被 丢弃 掉 。 通 过 使 用 单词 流 ， 先 命名 菜单 条 上 
的 顶级 菜单 ， 然 后 创建 子 菜单 ， 并 用 单词 来 填充 它 ， 直 至 没有 更 多 的 单词 位 置 。 
在 对 选取 菜单 项 的 响应 中 ，Listener 直 接 打 印 了 事件 ， 因 此 你 可 以 看 到 它 包含 什么 类 型 的 信 
息 。 当 你 运行 这 个 程序 时 ， 将 会 看 到 部 分 信息 包括 菜单 上 的 标签 ， 因 此 可 以 根据 这 些 信息 来 产 
生 对 不 同 菜单 的 响应 ， 或 者 你 可 以 为 每 个 菜单 都 提供 一 个 不 同 的 监听 器 (对 于 国际 化 来 说 ， 这 
是 一 种 更 安全 的 方式 )。 ， 
22.14.5 页 签 面板 、 按 钮 和 事件 
SWT 有 大 量 的 控件 集 ， 它 们 被 称 为 部 件 (widget) 。 可 以 浏览 org.eclipse.swt.widget 文 档 去 查 
看 基本 部 件 ， 以 及 浏览 org.eclipse.swt.custom 文 档 去 查看 更 奇特 的 部 件 。 
为 了 演示 各 种 基本 部 件 ， 下 面 的 程序 在 页 签 面板 内 部 放置 了 大 量 的 子 示例 。 你 还 将 看 到 如 
何 创建 Composite (大 体 与 Swing 的 JPanel 相 同 ) ， 以 实现 将 一 些 部 件 放 到 其 他 的 部 件 中 。 


//: swt/TabbedPane. java 
// Placing SWT components in tabbed panes. 
import swt.utit.*; 

import org.eclipse.swt.*; 

‘import org.eclipse.swt.widgets.*; 
import org.eclipse. swt.events.*; 
import org.eclipse. swt.graphics.* 
‘import org.eclipse.swt.layout.*; 
‘import org.eclipse.swt.browser.*: 





public class TabbedPane implements SWTApplication { 
private static TabFolder folder; 
private static Shell shell: 
Public void createContents(Composite parent) { 
shell = parent.getShel1(); 
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parent. setLayout (new FillLayout()); 
folder = new TabFolder (shell, SWT.BORDER): 
MabetTab() ; 
directoryDialogTab(); 
buttonTab(); 
sliderTab(): 
scribbleTab() ; 
browserTab(); 
$ 
public static void labelTab() { 
TabItem tab = new TabItem(folder, SWT.CLOSE); 
tab.setText("A Label"); // Text on the tab 
tab.setToolTipText("A simple label"); 
Label label = new Label(folder, SWT.CENTER); 
label.setText("Label text"); 
tab.setControl (label); 
} 
Public static void directoryDialogTab() { 
TabItem tab = new TabItem(folder, SWT.CLOSE) ; 
tab.setText("Directory Dialog"); 
tab. setToolTipText("Select a directory"); 
final Button b = new Button(folder, SWT.PUSH) ; 
b.setText("Select a Directory"); 
b.addListener(SWT.MouseDown, new Listener() { 
Public void handleEvent (Event e) { 
DirectoryDialog dd = new DirectoryDialog(shell); 
String path = dd.open(); 
if(path != null) 
b.setText (path) ; 
} 
H; 
tab. setControl (b) ; 
} 
public static void buttonTab() { 
TabItem tab = new TabItem(folder, SWT.CLOSE); 
tab, setText ("Buttons") ; 
tab. setToolTipText ("Different kinds of Buttons"); 
Composite composite = new Composite(folder, SWT.NONE); 
composite. setLayout(new GridLayout(4, true)); 
for(int dir : new int(}{ 
SWT.UP, SWT.RIGHT, SWT.LEFT, SWT.DOWN 
Ra 
Button b = new Button(composite, SWT.ARROW | dir); 
b.addListener (SWT.MouseDown, listener); 
} 
newButton(composite, SWT.CHECK, "Check button"); 
newButton(composite, SWT.PUSH, "Push button"); 
newButton (composite, SWT.RADIO, “Radio button"); 
newButton(composite, SWT.TOGGLE, "Toggle button"); 
newButton(composite, SWT.FLAT, “Flat button"): 
tab. setControl (composite) ; 
} 
private static Listener listener = new Listener() { 
Public void handleEvent (Event e) { 
MessageBox m = -new MessageBox (shell, SWT.OK): 
m,setMessage(e. toString()) ; 
m.open(); 
} 
i 
private static void newButton(Composite composite, 
int type, String label) { 
Button b = new Button(composite, type): 
b.setText(label); 
b.addListener(SWT.MouseDown, listener); 
} 
public static void sliderTab() { 
Tabltem tab = new TabItem(folder, SWT.CLOSE); 
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tab.setText("Sliders and Progress bars"); 
tab.setToolTipText ("Tied Slider to ProgressBar"): 
Composite composite = new Composite(folder, SWT.NONE) ; 
composite. setLayout (new GridLayout(2, true)); 
final Slider slider = 
new Slider(composite, SWT.HORIZONTAL) ; 
final ProgressBar progress = 
new ProgressBar (composite, SWT.HORIZONTAL) ; 
slider ..addSelectionListener(new SelectionAdapter() { 
public void widgetSelected(SelectionEvent event) { 
progress. setSelection(slider.getSelection()) ; 


} 
D: 
tab. setControl (composite) ; 
} 
public static void scribbleTab() { 
TabItem tab = new TabItem(folder, SWT.CLOSE): 
tab. setText("Scripble") ; 
tab.setToolTipText("Simple graphics: drawing”); 
final Canvas canvas = new Canvas(folder, SWT.NONE) : 
ScribbleMouseListener sml= new ScribbleHouseListener(); 
canvas. addMouseListener (sml) ; 
canvas. addMouseMovel tstener(sml) ; 
tab. setControl (canvas) ; 
} 
private static class ScribbleMouseListener 
extends HouseAdapter implements HouseMoveListener ( 
private Point p = new Point(®, 6); 
public void mouseHove (MouseEvent 
if((e.stateMask & SWT.BUTTONI) 
return; 
GC gc = new GC( (Canvas)e.widget) ; 
gc.drawLine(p.x, p.y, e.x, e.y); 
BC .dispose(); 
updatePoint (e) ; 
} 
Public void mouseDown(MouseEvent e) { updatePoint(e): } 
Private void updatePoint (MouseEvent e) { 
peX = exi 
py =e. 
} 
} 
Public static void browserTab() { 
TabItem tab = new TabItem(folder, SWT.CLOSE) ; 
tab. setText ("A Browser"); 
tab. setToolTipText ("A Web browser"): 
Browser browser = null; 
try { 
browser = new Browser(fotder, SWT.NONE) ; 
} catch(SWTError e) { 
Label label = new Label(folder, SWT.BORDER) : 
label. setText ("Could not initialize browser“); 
tab. setControl (label); 
) 
1f(browser != null) 《 
browser. setUrl("http://www.mindview.net"): 
tab. setControl (browser); 
} 








} 

public static void main(String{) args) { 
SwWTConsole.run(new TabbedPane(), 809; 680); 

$ 


YU 


这 里 ，createContents0 设 置 了 布局 ， 然 后 调用 了 一 些 方法 ， 每 个 方法 都 创建 了 一 个 不 同 的 
页 签 。 在 每 个 页 签 上 的 文本 都 是 用 setText0 设 置 的 (你 还 可 以 在 页 签 上 创建 按钮 和 图 形 )， 并且 
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每 个 页 签 还 都 设置 了 它 的 工具 提示 文本 。 在 每 个 方法 的 末尾 ， 你 将 看 到 对 setControl0 的 调用 ， 
这 个 方法 将 它 创建 的 控件 置 于 该 特定 页 签 的 对 话 框 空间 内 。 

labelTab0 演 示 了 一 个 简单 的 文本 标签 。directoryDialogTab() 持 有 一 个 按钮 ， 该 按钮 可 以 打 
开 一 个 标准 的 DirectoryDialog 对 象 ， 因 此 用 户 可 以 选择 一 个 目录 。 所 产生 的 结果 将 设置 为 这 个 
按钮 的 文本 。 

buttonTab(O) 展 示 了 不 同 的 基本 按钮 。sliderTab0 重 复 了 本 章 早先 的 Swing 示例 ， 将 一 个 滑 块 
与 进度 条 绑 定 在 一 起 。 

scribbleTab0 是 有 关 图 形 的 一 个 有 趣 的 示例 ， 一 个 绘图 程序 就 这 样 通过 数量 不 多 的 代码 行 构 
建 出 来 了 。 

最 后 ，browserTab() 展 示 了 SWT 的 Browser 组 件 的 威力 一 在 单个 组 件 中 的 一 个 全 功能 的 
Web 浏 览 器 。 
22.14.6 图 形 

下 面 是 将 Swing 程 序 SineWave.java 转 译 为 SWT 的 版 本 : 


1/; swt/SineWave. java 
11 SWT translation of Swing Sinewave. java. 
import swt.util.*; 

import org.eclipse.swt.*: 
import org.eclipse.swt.widgets 
import org.eclipse.swt.events. 
import org.eclipse. swt. layout. 





class SineDraw extends Canvas { 
private static final int SCALEFACTOR = 200; 
private int cycles; 
private int points; 
private doubleI] sines; 
private int[] pts; 
public SineDraw(Composite parent, int style) { 
super (parent, style): 
addPaintlistener(new PaintListener() { 
public void paintControl(Paintévent e) { 
int maxWidth = getSize().x; 
double hstep = (double)maxWidth / (double)points; 
int maxHeight = getSize().y; 
pts = new int{points); 
for(int 1 = @; i < points; i++) 
pts(i] = (int)((sines{i] * maxHeight / 2 + .95) 
+ (maxHeight / 2)): 
€. gc. set Foreground( 
e. display. getSystemColor (SWT.COLOR_RED)) : 
for(int 1 = 1; 1 < points; i++) { 
int xl = (int)((1 - 1) * hstep); 
int x2 = (int) (i * hstep); 
int yl = pts(i - 1]; 
int y2 = pts[i]; 
e.gc.drawLine(xl, yl, x2, y2); 
} 
} 


Di 
setCycles(5); 








} 
public void setCycles(int newCycles) { 
cycles = newCycles; 
points = SCALEFACTOR * cycles * 2: 
Sines = new double[points) ; 
for(int 1 = @; i < points: i++) { 
double radians = (Math.PI / SCALEFACTOR) * i; 
sines[i] = Math.sin(radians); 
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} 
redraw(); 
} 
} 


public class SineWave implements SWTApplication { 
private SineDraw sines; 
private Slider slider; 
public void createContents(Composite parent) { 
parent .setLayout (new GridLayout (1, true); 
sines = new SineDraw(parent, SWT.NONE) ; 
sines.setlLayoutData( 
new GridData(SWT.FILL, SWT.FILL, true, true)); 
sines.setFocus(); * 
slider = new Slider(parent, SWT.HORIZONTAL) ; 
slider.setValues(S, 1, 30, 1, 1, 1); 
slider. setLayoutData( 
new GridData(SWT.FILL, SWT.DEFAULT, true, false)): 
slider .addSelectionListener(new SelectionAdapter() { 
public void widgetSelected(SelectionEvent event) { 
sines. setCycles(slider.getSelection()); 
} 
Ds 
} 
public static void main(String[] args) { 
SWTConsole. run(new SineWave(), 766, 490); 


} 

} Mi~ 
与 JPanel 不 同 ， 在 SWT 中 基本 的 绘图 面 是 Canvas。 

如 果 对 这 个 版 本 的 程序 和 Swing 版 本 的 程序 进行 比较 ， 你 就 会 看 到 SineDraw 实 际 上 是 相同 
的 。 在 SWT 中 ， 你 可 以 从 提交 给 PaintListener 的 事件 对 象 中 获取 图 形 上 下 文 gc， 而 在 Swing 中 ， 
Graphics 对 象 被 直接 提交 给 了 paintComponent( 方 法 。 但 是 用 图 形 对 象 执行 的 行为 是 相同 的 ， 
而 且 setCycle0 是 相同 的 。 

ereateContents0 所 需 的 代码 比 Swing 版 的 要 稍微 多 一 点 ， 这 是 为 了 对 控件 布局 以 及 设置 滑 块 
及 其 监听 器 ， 但 是 ， 基 本 的 行为 仍旧 大 体 相同 。 

22.14.7 SWT 中 的 并 发 

尽管 AWT/Swing 是 单线 程 的 ， 但 是 如 果 产 生 非 确定 性 的 程序 ， 那 么 仍旧 有 可 能 违反 这 种 单 
线程 性 。 基 本 上 ， 你 不 会 想 要 用 多 线程 来 编写 显示 功能 ， 因 为 如 果 这 样 的 话 ， 它 们 就 会 以 令 人 
惊讶 的 方式 互相 改写 了 。 

SWT 根 本 不 允许 这 样 一 如 果 你 试图 用 多 个 线程 来 编写 显示 功能 ， 那 么 它 就 会 抛 出 异常 。 
这 可 以 防止 程序 员 新 手 因 不 注意 而 犯 此 类 错误 ， 从 而 在 程序 中 引入 难以 发 现 的 缺陷 。 

下 面 是 Swing 版 本 的 ColorBoxes.java 程 序 转移 为 SWT 的 版 本 : 


//: swt/ColorBoxes. java 
// SWT translation of Swing ColorBoxes. java. 
import swt.util.*; 

import org.eclipse.swt.*: 

import org.eclipse.swt.widgets.*; 

import org.eclipse.swt.events. 
import org.eclipse.swt.graphics.*; 
import org.eclipse. swt. layout.*; 
import java.util.concurrent.*; 
import java.util.*; 
import net.mindview.utit.*; 











class CBox extends Canvas implements Runnable { 
class CBoxPaintListener implements PaintListener { 
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public void paintControl(PaintEvent e) { 
Color color = new Color(e.display, cColor); 
e.gc.setBackground (color. 
Point size = getSize(); 
e.gc.fillRectangle(®, @, size.x, size.y): 
color.dispose(); 
} 
} 
private static Random rand = new Random(); 
private static RGB newColor() { 
return new RGB(rand.nextInt (255), 
rand.nextInt(255), rand.nextInt(255)); 





| } 
private int pause; 
private RGB cColor = newColor(); 
| Public CBox (Composite parent, int pause) { 
super (parent, SWT.NONE); 
this.pause = pause; 
addPaintListener (new CBoxPaintListener()); 
} 
public void run() { 
try { 
while(!Thread.interrupted()) { 
cColor = newColor(); 
getDisplay() .asyncExec (new Runnable() { 
public void run() { 
try { redraw(); } catch(SWTException e) {} 
// SWTException is OK when the parent 
// is terminated from under us. 
} 
D: 
TimeUnit MILLISECONDS. sleep (pause); 
} 
} catch(InterruptedException e) { 
// Acceptable way to exit 
} catch(SWTException e) { 
// Acceptable way to exit: our parent 
// was terminated from under us. 
} 
$ 
} 


public class ColorBoxes implements SWTApplication { 

1445 private int grid = 12; 

private int pause = 50; 

public void createContents (Composite parent) { 
GridLayout gridLayout = new GridLayout (grid. true); 
gridLayout .horizontalSpacing = 0; 
gridlayout.verticalSpacing = ©; 
parent. setLayout (gridLayout) ; 





ExecutorService exec = new DaemonThreadPoolExecutor(); 


for(int 1 = @; 1 < (grid * grid); i++) { 
final CBox cb = new CBox(parent, pause); 
cb.setLayoutData(new GridData(GridData.FILL_BOTH)); 
exec. execute (cb); 
} 
} 
public static void main(String(] args) { 
ColorBoxes boxes = new ColorBoxes(); 
if(args.length > 9) 
boxes.grid = new Integer (args [0]); 
if(args.length > 1) 
boxes. pause = new Integer(args(1]): 
SWTConsole.run(boxes, 568, 488): 
} 
} M~ 
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与 前 一 个 示例 一 样 ， 绘 制 是 通过 创建 带 有 paintControl0 方 法 的 PaintListener 来 控制 的 ， 这 个 方 
法 将 在 SWT 线 程 准 备 绘制 你 的 组 件 时 被 调用 。PaintListener 是 在 CBox 的 构造 器 中 注册 的 。 

CBox 的 这 个 版 本 明显 不 同 的 地 方 在 于 run0 方 法 ， 它 不 能 只 是 直接 调用 redraw0， 而 是 必须 
将 redraw() 提 交 给 Display 对 象 上 的 asyneExee() 方 法 ， 这 个 方法 大 体 上 与 SwingUtilities. 
invokeLater0 相 同 。 如 果 将 其 替换 为 对 redraw0 的 直接 调用 ， 就 会 看 到 程序 将 停 在 那里 。 

在 运行 这 个 程序 时 ， 你 会 看 到 少量 的 可 视 化 瑕 羔 ， 即 水 平 线 穿越 箱 体 。 这 是 因为 SWT 默 认 
情况 下 不 是 双 缓 存 的 ， 但 Swing 是 。 试 着 并 排 运 行 Swing 版 本 和 SWT 版 本 ， 你 就 会 看 得 更 清楚 。 
你 可 以 编写 双 缓存 的 SWT 代 码 ， 在 www.eclipse.org 网 站 可 以 找到 示例 。 

练习 42: (4) 修改 swUColorBoxesjava， 使 其 以 闪烁 点 (“星星”) 穿越 画布 开始 ， 然 后 随机 
地 改变 这 些 “ 星 星 ”的 颜色 。 

22.14.8 SWT 还 是 Swing 

在 如 此 短 的 介绍 中 很 难 了 解 全 貌 ， 但 是 你 至 少 开始 发 现 ， 在 许多 场合 ，SWT 是 一 种 比 Swing 
更 加 简单 的 编写 代码 方式 。 但 是 采用 SWT 的 GUI 编程 仍旧 很 复杂 ， 因 此 你 使 用 SWT 的 动机 可 能 
应 该 是 : 首先 ， 为 用 户 在 使 用 你 的 应 用 系统 时 提供 一 种 更 加 透明 的 体验 (因为 你 的 应 用 程序 的 
感官 与 在 用 户 平台 上 的 其 他 应 用 程序 相同 ) ， 其 次 ， 如 果 认 为 SWT 提 供 的 可 响应 性 很 重要 。 否 
则 ，Swing 就 可 能 是 恰当 的 选择 。 

练习 43: (6) 选择 任何 一 个 没有 在 本 节 转 译 的 Swing 示例 ， 将 其 转译 为 SWT。( 注 意 : 这 对 于 
培训 班 来 说 ， 是 一 个 很 好 的 家 庭 作 业 练习 ， 因 为 它 的 解决 方案 并 不 在 解决 方案 指南 中 。) 


22.15 总 结 


Java 的 GUI 类 库 在 Java 语 言 的 生命 周期 中 产生 了 翻天 覆 地 的 变化 。Java 1.0 的 AWT 被 批评 为 
一 种 差劲 的 设计 ， 它 允许 人 们 编写 可 移植 的 程序 ， 不 过 写 出 来 的 图 形 界面 也 是 “在 所 有 平台 上 
都 表现 一 般 "。 与 其 他 为 特定 平台 量 身 定做 的 应 用 开发 工具 相 比 ， 它 限制 很 多 ， 使 用 笨拙 ， 让 人 
难以 接受 。 

随 着 Java 1.1 引 入 了 新 的 事件 模型 和 JavaBean 规 范 ， 新 的 时 代 开 始 了 。 在 可 视 化 [DE 中 可 以 
很 容易 地 通过 拖 放 组 件 来 生成 程序 。 此 外 ， 事 件 模型 和 JavaBean 的 设计 清楚 地 表明 ， 设 计 者 在 
易于 编程 和 维护 代码 方面 (Java 1.0 的 AWT 里 看 不 出 这 些 ) 作 了 充分 考虑 。 但 是 直到 JFC/Swing 
出 现 ， 这 种 转变 才 算 完成 。 随 着 Swing 组 件 的 引入 ， 跨 平台 的 GUI 编 程 才 成 为 大 众 化 的 技术 。 

IDE 是 真正 革命 性 的 进步 。 要 是 你 希望 得 到 更 好 的 IDE 构 建 工具 ， 你 就 只 能 寄 希 望 于 提供 商 来 
满足 你 的 要 求 。 但 是 ，Java 作 为 一 个 开放 的 环境 ， 它 不 仅 允 许 IDE 构 建 工具 之 间 的 竞争 ， 而 且 鼓 励 
这 种 竞争 。 对 于 那些 “正规 的 ”工具 ， 它 们 必须 支持 JavaBean。 这 就 意味 着 一 个 公平 的 竞争 环境 ， 
如 果 有 更 好 的 工具 ， 你 不 必 拘泥 于 现 有 工具 。 你 可 以 采用 这 个 新 工具 提高 生产 率 。 对 于 GUI 的 IDE 
而 言 ， 这 种 新 式 的 竞争 环境 以 前 从 未 出 现 过 ， 结 果 也 必 将 对 程序 员 的 生产 率 产生 积极 的 影响 。 

本 章 的 目的 仅仅 是 向 读者 介绍 GUI 的 功能 ， 让 大 家 能 够 入 门 ， 这 样 就 可 以 在 使 用 库 的 过 程 
中 体会 到 Swing 的 方便 。 目 前 所 介绍 的 内 容 ， 对 于 设计 一 个 比较 好 的 用 户 界面 可 能 已 经 足够 了 。 
不 过 ，Swing、SWT 和 Flash/Flex 还 有 很 多 内 容 ， 它 们 的 设计 目标 是 要 成 为 一 个 功能 完整 的 UI 设 
计 工具 包 。 只 要 是 你 想到 的 ， 都 有 可 能 用 它 来 实现 。 

22.15.1 资源 

在 www.galbraiths.org/presentations 上 的 Ben Galbraith 的 在 线 演讲 提供 了 一 些 对 Swing 和 SWT 的 
精彩 介绍 。 

所 选 习题 的 答案 都 可 以 在 名 为 The Thinking in Java Annotated Solution Guide 的 电子 文档 中 找 
到 ,读者 可 以 从 www.MindView.net 处 购买 此 文档 。 
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附录 A 补充 材料 


本 书 还 备 有 大 量 的 补充 材料 ， 包 括 通过 MindView 网 站 提供 的 文集 、 讨 论 课 和 服务 。 
这 份 附录 是 对 这 些 补充 材料 的 说 明 ， 读 者 可 以 据 此 判断 它们 是 否 会 对 自己 有 帮助 。 


A.1 可 下 载 的 补充 材料 


本 书 的 代码 可 以 从 www.MindView.net 下 载 ， 其 中 包括 成 功 构建 并 执行 本 书 中 所 有 示例 所 必 
需 的 Ant 构 建文 件 和 其 他 支持 文件 。 

另外 ， 部 分 有 些 部 分 已 经 被 移入 到 电子 版 中 。 这 些 主题 包括 : 

“克隆 对 象 

“传递 和 返回 对 象 

"分 析 与 设计 

” 《Java 编程 思想 第 3 版 》 的 其 他 章节 中 一 些 相关 性 不 大 ， 不 应 该 放 到 本 书 第 4 版 中 的 部 分 。 


A.2 Thinking in C; Foundations for Java 


在 www.MindView.net， 你 将 发 现 可 以 免费 下 载 的 《Thinking in C》。 这 是 一 个 多 媒体 课件 ， 
由 Chuck Allison 创 建 ， 由 MindView 开 发 ， 可 以 给 你 关于 Java 语 法 所 基于 的 C 语 法 、 操 作 符 和 函数 
的 介绍 。 

注意 ,为 了 播放 《Thinking in C》， 你 必须 在 系统 上 安装 来 自 wwwMacroMediacom 的 Flash Player, 


A.3 Thinking in Java 讨 论 课 


我 的 公司 MindView 有 限 公司 提供 为 时 5 天 的 、 亲 身体 验 的 、 面 向 公众 的 室内 培训 讨论 课 ， 
其 内 容 基于 本 书 的 资料 。 它 以 前 的 名 称 是 “Hand-on Java” 讨论 课 , 这 是 我 们 主要 的 初级 讨论 课 ， 
能 够 为 学 习 更 高 级 的 讨论 课 打下 基础 。 每 次 课程 的 内 容 是 从 各 章 精 选 出 来 的 ， 课 程 之 后 是 在 指 
导 下 练习 ， 这 样 学 生 就 能 够 得 到 单独 指导 。 读 者 可 以 从 www.MindView.net 得 到 有 关 时 间 、 地 点 、 
推荐 书 及 其 他 详细 信息 。 


A.4 “Hand-on Java” 讨 论 课 CD 


“Hand-on Java” 讨 论 课 CD 包 含 了 “Thinking in Java” 讨 论 课 里 的 扩展 内 容 ， 同 时 它 也 基于 
本 书 的 资料 。 使 用 该 材料 ， 读 者 至 少 可 以 在 不 用 舟 车 劳顿 ， 并 能 够 减少 花费 的 情况 下 获得 生动 
的 培训 经 验 。 根 据 书 中 的 每 一 章 ， 它 附带 了 音频 讨论 课 和 相应 的 幻灯 片 。 我 创建 了 这 个 讨论 课 ， 
并 叙述 了 光盘 里 的 内 容 。 这 个 资料 是 Flash 格 式 的 ， 因 此 它 应 该 可 以 在 任何 支持 Flash Player 的 平 
台 上 运行 。“Hand-on Java” 讨 论 课 CD 可 以 在 wwwMindViewnet 购 买 ， 你 还 可 以 在 这 个 网 址 找到 
这 个 产品 的 试用 demo。 


A.5 Thinking in Objects 讨 论 课 
这 个 讨论 课 从 设计 者 的 角度 介绍 了 面向 对 象 编程 的 思想 。 它 探讨 了 开发 和 构建 系统 的 过 程 ， 
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主要 关注 于 所 谓 的 “敏捷 方法 ”或 者 “ 轻 量 级 方法 学 ”"， 尤 其 是 极限 编程 (XP)。 我 将 对 这 些 方 
法 学 作 总 体 介绍 ， 还 将 介绍 类 似 “ 索 引 卡 片 ”这 样 的 小 工具 ，“ 索 引 卡 片 ”是 在 Beck 和 Fowler 所 
著 的 《Planning Extreme Programming) (Addison-Wesley, 2001) 一 书 中 介绍 的 计划 编制 技术 ,还 
有 用 于 对 象 设计 的 CRC 卡 、 结 对 编程 、 选 代 计 划 、 单 元 测试 、 自 动 构建 、 源 代码 控制 ， 以 及 其 
他 类 似 主题 。 这 个 课程 还 包括 了 一 个 采用 XP 方 法 的 项 目 ， 将 在 一 周 内 开发 完成 。 

如 果 你 在 启动 一 个 项 目 ， 并 且 希 望 开始 使 用 面向 对 象 设计 技术 ， 那 么 我 们 可 以 用 你 的 项 目 
作为 示例 ， 在 一 周 结束 时 产生 一 个 第 一 稿 的 设计 。 

请 访问 www.MindView.net 得 到 有 关 时 间 、 地 点 、 推 荐 书 及 其 他 详细 信息 。 


A.6 《Thinking in Enterprise Java》 图 书 


本 书 从 《Thinking in Java》 中 部 分 讲述 高 级 主题 的 章节 派生 而 来 。 它 并 不 是 《Thinking in 
Java》 的 第 二 卷 ， 而 是 着 眼 于 企业 级 程序 设计 中 的 高 级 主题 。 这 本 书 现在 可 以 从 www.MindView. 
net (以 某 种 形式 ， 例 如 正 处 于 撰写 状态 ) 免费 下 载 。 由 于 是 一 本 单独 的 书 ， 因 此 它 的 篇 幅 可 以 
随 着 内 容 的 需要 而 扩展 。 与 《Thinking in Java) 一样 ， 它 的 目标 是 向 读者 提供 一 本 易于 理解 、 池 
盖 企 业 级 编程 技术 的 基础 读物 。 并 为 读者 学 习 更 深入 的 主题 做 准备 。 

以 下 列 出 的 是 书 中 讨论 的 部 分 主题 : 

“企业 级 程序 设计 介绍 

"使 用 Socket 和 Channel 进 行 网 络 编程 

"远程 方法 调用 (RMI) 

“连接 到 数据 库 

“命名 与 目录 服务 

* Servlet 

"Java 服 务 器 页 面 (JSP) 

“标签 、JSP 片段 和 表示 语言 

* 自动 产生 用 户 界面 

。 企 业 级 Java Beans (EJB) 

“可 扩展 标记 语言 (XML) 

。Web 服 务 

“自动 测试 

可 以 从 wwwMindView net 网 站 上 了 解 《Thinking in Enterprise Java》 的 进展 情况 。 


A.7 《Thinking in Patterns (with Java)》 图 书 


面向 对 象 设计 领域 的 重大 进步 之 一 ， 就 是 “设计 模式 ”运动 的 兴起 ，Gamma, Helm, Johnson 
& Vlissides (Addison-Wesley 1995) 的 《Design Patterns) 一 书 对 此 有 描述 。 这 本 书 介绍 了 23 种 
不 同 的 解决 方案 ， 它 们 都 专门 针对 某 些 特定 类 型 的 问题 ， 并 以 C++ 语言 表述 。《 设 计 模式 》 已 经 
成 为 经 典 之 作 ， 它 是 面向 对 象 程序 员 之 间 进 行 交流 的 通用 词汇 ， 这 几乎 就 成 了 一 种 规定 。 
《Thinking in Patterns》 介 绍 了 设计 模式 的 基本 概念 ， 并 附 有 Java 范 例 。 这 本 书 并 不 希望 成 为 《 设 
计 模式 》 的 简单 重复 ， 而 是 希望 从 Java 角 度 带 来 新 的 观点 。 它 并 不 仅 限于 传统 的 23 种 模式 ， 还 
包括 了 其 他 一 些 恰当 的 问题 解决 方案 。 

这 本 书 脱胎 于 《Thinking in Java》 第 1 版 的 最 后 一 章 ， 随 着 内 容 的 不 断 发 展 ， 把 它 单独 成 书 
就 显得 合情合理 。 在 编写 这 份 附录 的 时 候 ，《Thinking in Patterns》 还 在 写作 之 中 ， 但 其 中 的 素材 
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已 经 无 数 次 在 “Objects & Patterns” 讨 论 课 的 实践 中 使 用 过 〈 它 现在 已 经 被 划分 为 “Designing 
Objects & Systems” 讨 论 课 和 “Thinking in Pattems” 讨 论 课 ) 。 
在 www.MindView.net 处 ， 你 可 以 发 现 有 关 这 本 书 的 更 多 内 容 。 


A.8 Thinking in Patterns 讨 论 课 


本 讨论 课 从 “Objects & Patterms” 讨 论 课 衍生 而 来 ，Binl Venners 和 我 在 过 去 儿 年 一 直 在 开 那 
个 讨论 课 。 但 它 的 内 容 越 来 越 多 ， 所 以 我 们 将 它 分 成 两 个 : 这 一 个 和 本 附录 前 面 说 明 的 
“Designing Objects & Systems” 讨 论 课 。 

本 讨论 课 严 格 遵循 《Thinking in Patterns》 一 书 里 的 资料 和 表述 ， 所 以 要 了 解 本 讨论 课 的 内 
容 ， 最 好 是 从 www.MindView.net 下 载 这 本 书 。 

许多 表述 都 是 设计 演化 过 程 的 实例 ， 先 从 初始 解决 方案 开始 ， 然 后 通过 演化 过 程 ， 得 到 更 
恰当 的 设计 。 共 中 的 最 后 一 个 案例 (垃圾 回收 模拟 ) 已 经 随 着 时 间 而 演化 ， 读 者 可 以 把 这 个 演 
化 过 程 作为 一 个 原型 ， 这 样 你 的 设计 在 开始 的 时 候 就 对 这 类 特定 问题 有 了 足够 的 思路 ， 然 后 对 
这 一 类 问题 演化 出 灵活 的 方案 。 

* 极 大 增强 设计 的 灵活 度 。 

* 内置 的 可 扩展 性 和 可 重用 性 。 

“使 用 模式 语言 进行 设计 之 间 的 交流 。 

每 次 课程 之 后 将 有 一 些 模式 练习 有 待 解决 ， 这 些 练习 将 引导 你 编写 代码 ， 应 用 特定 的 模式 ， 
从 而 得 到 编程 问题 的 解决 方案 。 

请 访问 www.MindView.net 得 到 有 关 时 间 、 地 点 、 推 荐 书 及 其 他 详细 信息 。 

《设计 模式 》 中 文 版 、 英 文 版 及 双语 版 均 已 由 机 械 工 业 出 版 社 出 版 。 


A.9 设计 咨询 与 评审 


我 的 公司 还 以 提供 咨询 、 辅 导 、 设 计 评审 和 实现 评审 的 方式 ， 在 项 目的 整个 开发 周期 为 你 
提供 帮助 ， 它 对 你 的 首 个 Java 项 目 尤 具 价 值 。 请 访问 www.MindView.net 以 获得 详细 信息 。 
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B.1 软件 


从 http://iava.sun.com 获 得 的 JDK (Java 开 发 工具 包 )。 即 使 你 选择 了 第 三 方 开发 环境 ， 万 一 
遇 到 了 可 能 是 编译 器 出 错 的 情况 ， 手 头 有 一 套 IDK 总 是 不 错 的 。 可 以 把 JDK 作 为 检验 标准 ， 因 为 
如 果 JDK 有 错误 ， 那 么 这 个 错误 广为人知 的 机 会 也 应 该 比较 高 。 

从 httpWiava.sun.com 获 得 HTML 格式 的 JDK 文 档 。 我 所 见 过 的 介绍 标准 Java 库 的 参考 书 不 是 
内 容 过 时 ， 就 是 有 所 和 遗漏。 尽管 Sun 的 这 份 HTML 文 档 有 不 少 小 错误 ， 而 且 有 时 过 于 简陋 ， 不 过 
它 至 少 列 出 了 所 有 的 类 和 方法 。 对 使 用 在 线 资源 而 不 是 印刷 书籍 ， 人 们 开始 可 能 会 有 些 不 习惯 ， 
不 过 克服 这 一 点 相当 值得 。 先 浏览 一 下 HTML 文 档 ， 至 少 你 可 以 得 到 大 概 的 印象 。 如 果 做 不 到 
这 一 点 ， 就 去 找 一 本 印刷 书籍 吧 。 


B.2 编辑 器 和 IDE 


在 这 个 竞技 场 上 有 着 健康 的 竞争 。 许 多 提供 的 产品 都 是 免费 的 不 免费 的 也 都 有 免费 的 试 
用 版 )， 因 此 最 好 的 办 法 就 是 自己 去 试验 它们 ， 来 看 看 哪个 更 适合 你 的 需求 。 下 面 是 其 中 的 一 
些 ， 

JEdit，Slava Pestov 的 免费 编辑 器 ， 用 Java 编 写 的 ， 因 此 你 可 以 获得 一 个 好 处 ， 就 是 可 以 看 
到 一 个 桌面 Java 应 用 在 运行 。 这 个 编辑 器 是 典型 的 基于 插件 的 软件 ， 许 多 插件 都 是 由 活跃 的 社 
区 编写 的 。 可 以 从 wwjedit.org 下 载 。 

NetBeans，Sun 的 免费 IDE， 位 于 www.netbeans.org。 设 计 用 于 拖 电 式 GUI 构建 、 代 码 编辑 、 
调试 和 共 他 目的 。 

Eclipse，IBM 支 持 的 开源 项 目 之 一 。Eclipse 平 台 还 被 设计 成 一 个 可 扩展 的 基础 ， 因 此 你 可 
以 在 Eclipse 之 上 构建 自己 单独 的 应 用 。 这 个 项 目 创建 了 在 第 22 章 中 描述 的 SWT。 可 以 从 
www.Eclipes.org 下 载 。 

IntelliJ IDEA， 大 量 的 Java 程 序 员 都 非常 喜欢 的 付费 软件 ， 许 多 程序 员 都 声称 ，IDEA 总 是 
比 Eclipse 快 一 两 步 ， 可 能 是 因为 Intellil 没 有 在 同时 创建 IDE 和 开发 平台 ， 而 是 坚持 专攻 IDE。 可 
以 从 wwwjetbrains.com 处 下 载 免费 试用 版 。 


B.3 书籍 


(Effective Java™) Joshua Bloch 著 (Addison-Wesley,2001), 希望 订正 Java 集 合 类 库 问 题 的 人 
手中 必 备 的 书籍 ， 按 照 Scott Meyer 的 经 典 著作 《Effective C++》 的 模式 编写 的 。 

{Core Java 2, 7* Edition, Volumes I&II》eHorstmann & Comell 著 (Prentice-Hall, 2005)。 这 两 本 
书 巨大 且 全 面 。 每 当 我 需要 寻找 某 些 答案 时 ， 就 会 想到 它们 。 当 你 读 完 《Thinking in Java) AF 
要 更 进一步 时 ， 我 推荐 这 两 本 书 。 

O 本 书 中 文 版 已 由 机 械 工业 出 版 社 出 版 。 一 编辑 注 

© 本 书 中 文 版 已 由 机 械 工业 出 版 社 出 版 。 一 -编辑 注 
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«The Java™ Class Libraries: An Annotated Reference), Patrick Chan 和 Rosanna Lee 著 
(Addison-Wesley, 1997) 。 尽 管内 容 有 些 过 时 ， 但 这 是 你 应 该 拥有 的 JDK 参 考 书 : 详细 的 说 明令 
其 使 用 起 来 非常 方便 。 这 本 书 很 庞大 且 昂 贵 ， 其 中 提供 的 示例 品质 并 不 能 令 我 满意 。 不 过 你 要 
是 遇 到 某 个 疑难 问题 ， 这 本 书 能 提供 比 其 他 可 供 选 择 的 书籍 更 深入 〈 也 更 详细 ) 的 解答 。 但 是 ， 
(Core Java 2》 对 许多 类 库 构建 有 最 新 的 覆盖 。 

Design Pattems) © ，Gamma、Helm、Johnson 和 Vlissides 著 (Addison-Wesley, 1995) 。 在 程 
序 设计 领域 发 起 设计 模式 运动 的 开山 之 作 ， 在 本 书 中 很 多 地 方 都 提 到 过 。 

«Refactoring to Patterns) © Joshua Kerievsky 著 (Addison-Wesley,2005)。 将 重 构 和 设计 模式 联 
姻 ， 这 本 书 的 最 可 取 之 处 就 是 它 展示 了 你 可 以 如 何 通过 引入 模式 来 演化 一 个 设计 。 

(The Art of UNIX Programming} Eric Raymond 著 (Addison-Wesley,2004)。 尽 管 Java 是 跨 平 台 
的 ,但 是 Java 在 服务 器 上 的 流行 使 得 对 Unix/Linux 有 所 了 解 变 得 很 重要 。Eric 的 书 是 对 这 种 操作 
系统 的 历史 和 哲学 的 一 个 极 优秀 的 介绍 ， 并 且 ， 如 果 你 只 是 想 理解 计算 的 某 些 根基 性 的 知识 ， 
它 也 是 一 本 令 人 着 迷 的 读物 。 

B.3.1 分 析 与 设计 

Extreme Programming Explained, 2”Edition》，Kent Beck 著 (Addison-Wesley, 2005), 我 总 
觉得 应 该 会 有 与 众 不 同 、 更 好 的 软件 开发 过 程 ， 我 认为 XP 已 经 很 接近 这 个 标准 了 。 另 一 本 对 我 
有 同样 震撼 的 书 是 《Peopleware》 (后面 介绍 )， 它 主要 探讨 环境 和 团队 文化 中 的 协作 。《KExtreme 
Programming Explained》 探 讨 的 是 程序 设计 ， 它 要 推翻 为 众人 所 知 的 绝 大 多 数 方法 ， 其 至 是 最 
新 的 “研究 发 现 "。 书 中 的 叙述 甚至 非常 激进 ， 声 称 任何 有 关 项 目的 全 景 描述 只 要 没有 花费 你 太 
多 的 时 间 ， 而 且 你 愿意 将 它们 丢掉 ， 那 么 它们 就 是 好 的 选择 (你 会 注意 到 这 本 书 的 封面 上 没有 
“UML 认 证 标志 ")。 我 会 以 某 家 公司 是 否 采用 XP 来 决定 是 否 为 他 们 工作 。 这 本 书 短小 精 悍 ， 章 
节 很 短 ， 读 起 来 很 轻松 ， 而 且 能 够 激励 你 思考 。 你 可 以 开始 想象 自己 工作 在 这 样 的 环境 中 ， 它 
会 带 给 你 全 新 的 视野 。 

(UML Distilled, 2" Edition), Martin Fowler 著 (Addison-Wesley, 2000)。 初 次 接触 UML 时 大 
概 会 有 县 难 情绪 ， 因 为 里 面 充满 了 各 种 图 和 细节 。 根 据 Eowler 的 说 法 ， 其 实 大 部 分 内 容 都 非 必 
要 ， 所 以 他 直接 讨论 本 质 内 容 。 对 大 多 数 项 目 来 说 ， 你 只 要 把 少数 几 种 图 作为 工具 就 够 了 。 
Fowler 关 注 的 是 拿 出 一 份 好 的 设计 ， 而 不 是 要 得 到 这 一 份 好 设计 所 需要 的 全 部 制品 。 这 是 一 本 
优秀 、 短 小 精 悍 、 易 于 阅读 的 书籍 ， 如 果 你 需要 理解 UML， 这 本 书 是 首选 。 

《Domain-Driven Design》，Eric Evans 著 (Addison-Wesley, 2004)。 本 书 聚 焦 于 设计 阶段 的 主 
要 制品 : 域 模 型 。 我 发 现 本 书 是 一 种 重要 的 手段 ， 强 调 要 帮助 程序 员 保持 正确 的 抽象 级 别 。 

(The Unified Software Development Process) ® , Ivar Jacobsen, Grady Booch 和 James Rumbaugh 
著 (Addison-Wesley, 1999)。 我 原本 做 好 了 不 喜欢 这 本 书 的 打算 ， 此 书 似乎 具有 烦人 的 大 学 教科 
书 才 有 的 所 有 特征 。 但 是 我 惊喜 地 发 现 ， 全 书 不 仅 脉络 清晰 ， 而 且 令 人 愉快 。 尽 管 书 中 有 几 个 概 
念 似乎 作者 也 不 甚 明了。 其 中 最 好 的 一 点 是 ， 整 个 过 程 非常 具有 实用 价值 。 它 不 是 XP (而 且 没有 
它们 那样 清晰 的 测试 )， 但 它 也 是 UML 组 成 部 分 。 即 使 你 无 法 接受 XP， 但 在 大 多 数 人 已 经 认可 了 
“UML 就 是 好 ”( 且 不 论 他 们 实际 经 验 如 何 ) 的 情况 下 ， 你 也 许 会 接受 本 书 。 我 认为 此 书 应 当 是 
推广 UML 的 旗舰 。 当 你 读 完 Fowler 的 《UML Distiled》， 还 准备 深入 学 习 的 话 ， 可 以 选择 这 本 书 。 


O 本 书 中 文 版 、 英 文 版 影印 及 双语 版 已 由 机 械 工 业 出 版 社 出 版 。 一 一 编辑 注 
O 本 书 英文 影印 版 已 由 机 械 工业 出 版 社 出 版 。 一 -编辑 注 
© 本 书 中 文 版 已 由 机 械 工业 出 版 社 出 版 。 一 一 编辑 注 
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在 选择 任何 方法 之 前 ， 先 听 听 立场 中 立 人 士 的 看 法 会 很 有 帮助 。 人 们 往往 在 尚未 真正 了 解 
自己 的 需要 ， 或 尚未 知道 某 种 方法 能 为 你 做 什么 之 前 ， 就 轻率 地 作出 选择 。“ 别 人 正在 使 用 ”， 
听 起 来 似乎 很 有 道理 。 不 过 ， 人 们 常 有 一 种 奇怪 心理 :如 果 他 们 想 要 相信 某 种 方法 真能 解决 问 
题 ， 他 们 就 会 去 尝试 (这 种 实验 态度 很 好 )， 但 如 果 不 能 解决 问题 ， 他 们 便 可 能 加 们 努力 并 开始 
大 声 宣称 ， 他 们 发 现 了 很 伟大 的 东西 (这 种 拒绝 承认 的 态度 不 好 )。 这 里 的 假设 是 ， 如 果 有 一 些 
人 和 你 在 同一 艘 船上 ， 你 就 不 会 感到 孤单 ， 哪 怕 那 艘 船 正 驶 向 未 知 的 地 方 (其 至 正在 下 沉 )。 

我 并 不 是 在 说 所 有 方法 学 都 没有 前 途 ， 而 是 提醒 你 应 该 用 某 种 理念 来 武装 自己 ， 这 种 理念 
能 够 帮助 你 坚持 实验 模式 (“这 种 方法 不 可 行 ， 让 我 们 试 试 其 他 方法 ")， 并 摆脱 否认 模式 (“不 ， 
这 其 实 不 是 问题 。 一 切 都 是 那么 美好 ， 我 们 不 需要 改变 ") 。 我 认为 在 你 选择 某 种 方法 之 前 ， 应 
该 先 阅读 下 列 几 本 书 ， 它 们 会 带 给 你 这 种 理念 。 

Software Creativity), Robert L.Glass 著 (Prentice Hall, 1995) 。 在 完整 地 从 方法 论 角度 进行 讨论 
的 书籍 中 ， 这 是 我 见 过 最 好 的 一 本 。 本 书 集合 了 Glass 所 撰写 或 获得 (PJ. Plauger 是 其 中 一 位 作者 ) 
的 许多 小 品 文 和 论文 ， 这 些 文章 反映 出 他 多 年 来 对 这 个 课题 的 思考 和 研究 。 这 些 文章 十 分 有 趣 ， 
而 且 长 度 适中 ， 既 非 漫 无 目的 ， 也 不 会 让 你 感到 无 聊 。 作 者 也 不 是 毫 无 根据 ， 其 中 引用 了 数 以 百 
计 的 其 他 论文 和 研究 报告 。 所 有 程序 员 和 管理 者 在 陷入 方法 论 的 泥沼 前 ， 都 应 该 好 好 阅读 这 本 书 。 

«Software Runaways: Monumental Software Disasters}, Robert L.Glass 著 (Prentice Hall, 1998)。 
这 本 书 最 出 色 的 地 方 是 ， 它 直接 把 我 们 带 到 以 前 从 未 讨论 过 的 软件 开发 的 最 前 沿 ， 有 多 少 项 目 
不 仅 失败 了 ， 而 且 是 一 败 涂 地 。 我 发 现 大 多 数 人 仍然 认为 “这 不 可 能 发 生 在 我 身上 ”, 或 “这 不 
会 重演 "， 这 种 侥幸 心理 会 使 我 们 处 于 劣势 。 要 把 “任何 事 都 可 能 出 错 ”牢记 在 心 ， 这 样 才能 以 
更 好 的 心态 使 事情 向 正确 的 方向 发 展 。 

《Peopleware, 2”Edition》，Tom DeMarco 和 Timothy Lister 著 (Dorset House, 1999)。 这 是 必 读 
书 。 它 不 仅 有 趣 ， 而 且 会 动摇 你 的 世界 观 ， 挫 毁 你 不 切实 际 的 假设 。 虽 然 书 中 的 背景 是 软件 开 
发 ， 但 其 讨论 的 内 容 适 用 于 一 般 项 目 和 团队 。 其 重点 放 在 人 及 人 的 需求 ， 而 不 是 技术 和 技术 的 
需求 上 。 作 者 所 讨论 的 是 如 何 建立 一 个 让 人 们 能 够 快乐 工作 并 且 有 高 生产 率 的 环境 ， 而 不 是 讨 
论 这 些 人 应 该 遵守 哪些 规则 才能 成 为 称职 的 “机 器 零件 "。 我 认为 正 是 后 一 种 态度 造成 了 程序 员 
在 采用 某 种 方法 的 时 候 先 拍手 叫好 ， 然 后 并 不 做 出 任何 改变 。 

(Secrets of Consulting: A Guide to Giving & Getting Advice Successfully), Gerald M. Weinberg 
% (Dorset House, 1985)。 一 本 很 棒 的 书 ， 我 最 喜欢 的 书籍 之 一 。 如 果 你 准备 当 一 名 顾问 ， 或 者 
想 与 顾问 合作 愉快 ， 请 选择 本 书 。 书 中 的 章节 很 短 ， 里 面 有 很 多 故事 和 轶 闭 ， 它 们 引导 你 如 何 
付出 最 小 的 代价 来 看 清 问题 的 实质 。 也 可 以 参考 《More Secrets of Consulting》(2002 年 出 版 ) 或 
其 他 Weinberg 写 的 作品 。 

《Complexity》，M. Mitchell Waldrop 著 (Simon & Schuster, 1992)。 这 本 书记 录 了 一 群 来 自 不 
同 领域 的 科学 家 ， 聚 集 于 新 星 西 哥 州 圣 达 菲 (Santa Fe) ， 一 起 讨论 他 们 各 自学 科 领 域 无 法 解决 
的 现实 问题 (经济 学 里 的 股市 问题 ， 生 物 学 里 的 生命 起 源 问题 ， 社 会 学 里 的 人 类 行为 问题 ， 等 
等 )。 赁 借 跨 物理 、 经 济 、 化 学 、 数 学 、 计 算 机 科学 、 社 会 学 以 及 其 他 学 科 的 方式 ， 针 对 这 些 问 
题 发 展 出 一 套 学 科 交 又 的 解决 方案 。 更 重要 的 是 ， 思 考 这 类 极 复杂 问题 的 另 一 种 方式 正在 成 形 : 
抛弃 “数学 决定 论 ” 和 “以 方程 式 预测 所 有 行为 ”的 错误 认 知 ， 迈 向 “ 先 观 察 ， 找 出 模式 ， 试 
着 以 任何 可 能 的 手段 仿真 ”的 方式 。 比 如 ， 书 中 记录 了 遗传 算法 的 面世 。 我 相信 ， 这 种 思考 方 
式 对 我 们 研究 和 管理 日 益 复杂 的 软件 项 目 十 分 有 用 。 

B.3.2 Python 
(Learning Python, 2™ Edition), Mark Lutz 和 David Ascher  (O’ Reilly, 2003)。 一 本 针对 
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程序 员 的 人 门 读物 ， 也 是 我 最 喜欢 的 程序 语言 ， 和 Java 配 合 效果 更 好 。 本 书 还 包括 对 Jython 的 
介绍 。 使 用 Jython， 可 以 将 Java 和 Python 整合 进 同一 个 程序 (Jython 解 释 器 能 产生 Java 字 节 码 ， 
所 以 不 用 加 入 任何 特殊 操作 就 可 以 达到 目的 ) 。 这 个 语言 的 相关 组 织 承诺 将 为 我 们 带 来 最 大 的 
可 能 性 。 

B.3.3 我 的 作品 

以 下 书籍 并 非 目 前 都 能 找到 ， 但 是 有 些 可 以 在 二 手书 店 看 到 。 

(Computer Interfacing with Pascal & C》(1988 年 通过 Eisys 自 己 印 刷 。 只 能 通过 
www.BruceEckel.com 取 得 )。 这 是 在 CPM 为 主流 而 DOS 正 在 崛起 的 时 代 ， 一 本 带 有 电子 学 背景 的 
人 门 书 。 我 使 用 高 级 语言 通过 计算 机 并 行 端口 进行 控制 ， 来 驱动 各 种 电子 设备 。 本 书 内 容 改写 
自我 最 初 (也 是 最 好 的 ) 在 《Micro Cornucopia》 杂 志 上 发 表 的 专栏 文章 。 编 写 这 本 书 给 我 带 来 
了 极 好 的 出 版 经 验 。 

Using C++》(Osbome/McGraw-Hill, 1989) 。 我 的 第 一 本 C++ 书籍 。 本 书 已 经 绝版 ， 被 其 第 
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generic methods - 398, 635 

exponential notation - 109 

extending a class during inheritance - 35 

extends - 226, 243, 307; and @interface - 
1070; and interface - 330; keyword - 241 

extensible program - 286 

extension: sign - 112; zero - 112 

extension, vs. pure inheritance - 306 

Externalizable - 986; alternative approach 
to using - 992 

Extreme Programming (XP) . 1457 
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Façade - 577 

Factory Method design pattern - 339, 582, 
627, 928; and anonymous classes . 361 

factory object - 285, 664 

fail fast containers - 888 

false - 105 

FeatureDescriptor - 1414 

Fibonacci - 629 

Field, for reflection - 589 

fields, initializing fields in interfaces . 335 

FIFO (first-in, first out) - 423 

file: characteristics of files - 912; dialogs - 
1368; File class - 901, 916, 925; 
File.list( ) - 901; incomplete output files, 
errors and flushing + 931; JAR file - 212; 
locking - 970; memory-mapped files - 
966 


FileChannel - 947 

FileDescriptor - 916 

FileInputReader - 927 

FileInputStream - 916 

FileLock - 971 

FilenamefFilter . 901 

FileNotFoundException - 485 

FileOutputStream - 917 

FileReader . 483, 923 

FileWriter - 923, 930 

fillStackTrace( ) - 461 

FilterInputStream - 916 

FilterOutputStream - 917 

FilterReader - 924 

FilterWriter - 924 

final - 316, 622; and efficiency - 271; and 
private - 268; and static - 263; argument 
+ 266, 904; blank finals - 265; classes - 
270; data - 262; keyword - 262; method - 
282; methods . 267, 303; static 
primitives - 264; with object references . 
263 

finalize( ) - 173, 254, 485; and inheritance - 
295; calling directly - 175 

finally - 251, 254; and constructors . 483; 
and return . 476; keyword - 471; not run 
with daemon threads - 1135; pitfall . 477 

finding .class files during loading - 214 

FixedThreadPool - 1122 

flag, using EnumSet instead of - 1028 

Flex: OpenLaszlo alternative to Flex . 1416; 
tool from Macromedia - 1416 

flip(), nio - 948 

float: floating point true and false - 106; 
literal value marker (F) - 109 

FlowLayout - 1318 

flushing output files - 931 

Flyweight design pattern - 800, 1301 

focus traversal - 1305 
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folding, constant - 262 

for keyword - 138 

foreach - 141, 145, 199, 200, 219, 376, 393, 
422, 429, 545, 629, 631, 694, 1011, 
1036; and Adapter Method - 434; and 
Tterable - 431 

format: precision - 517; specifiers - 516; 
string . 514; width - 516 

format() +514 

Formatter - 515 

forName( ) - 558, 1326 

FORTRAN programming language - 110 

forward referencing . 184 

Fowler, Martin - 209, 495, 1457 

framework, control framework and inner 
classes - 375 

function: member function - 29; overriding, 
+36 

function object - 737 

functional languages - 1113 

Future . 1125 
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garbage collection - 173, 175; and cleanup - 
251; how the collector works - 178; order 
of object reclamation - 254; reachable 
objects . 889 

Generator - 285, 627, 636, 645, 695, 732, 
763, 780, 794, 1021, 1042; filling a 
Collection - 636; general purpose - 637 

generics: @Unit testing - 1094; and type- 
safe containers - 390; anonymous inner 
classes - 645; array of generic objects - 
850; basic introduction - 390; bounds - 
653, 673; cast via a generic class - 699; 
casting - 697; Class references - 565; 
contravariance - 682; curiously 
recurring - 702; erasure + 650, 696; 
example of a framework . 1282; 
exceptions - 711; explicit type argument 
specification for generic methods . 398, 
635; inner classes - 645; instanceof - 
663, 697; isInstance( ) - 663; methods - 
631, 795; overloading - 699; reification - 
655; self-bounded types - 701; simplest 
class definition - 413; supertype 
wildcards - 682; type tag - 663; 
unbounded wildcard - 686; varargs and 
generic methods - 635; wildcards - 677 

get( ): ArrayList - 390; HashMap - 420; no 
get( ) for Collection - 811 

getBeanInfo( ) . 1398 

getBytes( ) - 929 

getCanonicalName( ) - 560 

getChannel( ) - 948 

getClass( ) - 459, 558 


getConstructor( ) - 1335 

getConstructors( ) - 592 

getenv( ) . 433 

getEventSetDescriptors( ) . 1401 

getInterfaces( ) - 560 

getMethodDescriptors( ) - 1401 

getMethods( ) . 592 

getName( ) - 1400 

getPropertyDescriptors( ) - 1400 

getPropertyType( ) - 1400 

getReadMethod( ) - 1400 

getSelectedValues( ) - 1347 

getSimpleName( ) . 560 

getState( ) . 1357 

getSuperclass( ) - 561 

getWriteMethod( ) - 1400 

Glass, Robert - 1458 

glue, in BoxLayout - 1321 

Goetz Test, for avoiding synchronization - 
1160 

Goetz, Brian - 1156, 1160, 1272, 1302 

goto, lack of in Java - 146 

graphical user interface (GUI) . 375, 1303 

graphics - 1368; Graphics class . 1361 

greater than (>) - 103 

greater than or equal to (>=) - 103 

greedy quantifiers . 529 

GridBagLayout - 1320 

GridLayout - 1319, 1392 

Grindstaff, Chris - 1430 

group, thread - 1146 

groups, regular expression . 534 

guarded region, in exception handling - 
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GUI: graphical user interface . 375, 1303; 
GUI builders - 1304 
GZIPInputStream - 973 
GZ1POutputStream + 973 





handler, exception . 448 

Harold, Elliotte Rusty - 1415, 1456; XOM 
XML library - 1003 

has-a - 32; relationship, composition . 258 

hash function - 847 

hashCode( ) - 833, 839, 847; and hashed 
data structures - 843; equals( ) - 822; 
issues when writing . 851; recipe for 
generating decent - 853 

hashing - 844, 847; and hash codes . 839; 
external chaining . 848; perfect hashing 
function - 848 

HashMap - 834, 877, 1287, 1331 

HashSet - 415, 821, 872 

Hashtable - 877, 895 

hasNext( ), Iterator - 407 
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Hexadecimal - 109 

hiding, implementation - 228 
Holub, Allen . 1295 

HTML on Swing components . 1370 





T/O: available( ) - 930; basic usage, 
examples - 927; between tasks using 
pipes . 1221; blocking, and available( ) - 
930; BufferedInputStream . 920; 
BufferedOutputStream - 921; 
BufferedReader . 483, 924, 927; 
BufferedWriter - 924, 930; 
ByteArrayInputStream - 916; 
ByteArrayOutputStream , 917; 
characteristics of files - 912; 
CharArrayReader - 923; 
CharArrayWriter - 923; 
CheckedInputStream - 973; 
CheckedOutputStream - 973; close( ) - 
928; compression library - 973; 
controlling the process of serialization - 
986; DataInput - 926; DatalnputStream 
+ 920, 924, 929; DataOutput - 926; 
DataOutputStream - 921, 925; 
DeflaterOutputStream - 973; directory 
lister - 902; directory, creating 
directories and paths . 912; 
Externalizable . 986; File . 916, 925; File 
class . 901; File.list() . 901; 
FileDescriptor - 916; FileInputReader - 
927; FileInputStream - 916; 
FilenamefFilter . 901; FileOutputStream 
- 917; FileReader - 483, 923; FileWriter - 
923, 930; FilterInputStream - 916; 
FilterOutputStream - 917; FilterReader - 
924; FilterWriter - 924; from standard 
input - 941; GZIPInputStream - 973; 
GZIPOutputStream - 973; 
InflaterInputStream - 973; input - 914; 
InputStream - 914; InputStreamReader - 
922, 923; internationalization - 923; 
interruptible - 1189; library - 901; 
lightweight persistence - 980; 
LineNumberInputStream - 920; 
LineNumberReader - 924; mark( ) - 
926; mkdirs( ) - 914; network I/O - 946; 
new nio - 946; ObjectOutputStream - 
981; output - 914; OutputStream - 914, 
917; OutputStreamWriter - 922, 923; 
pipe - 915; piped streams - 936; 
PipedInputStream - 916; 
PipedOutputStream - 916, 917; 
PipedReader - 923; PipedWriter - 923; 
PrintStream - 921; PrintWriter - 924, 
930, 932; PushbackInputStream - 920; 
PushbackReader . 924; 








RandomAccessFile . 925, 926, 934; 
read( ) - 914; readDouble( ) - 934; 
Reader - 914, 922, 923; readExternal( ) - 
986; readLine( ) - 485, 924, 931, 942; 
readObject( ) - 981; redirecting standard 
1/0 - 942; renameTo( ) - 914; reset() + 
926; seek( ) - 926, 934; 
SequenceInputStream . 916, 925; 
Serializable - 986; setErr(PrintStream) - 
943; setIn(InputStream) - 943; 
setOut(PrintStream) : 943; 
StreamTokenizer - 924; StringBuffer - 
916; StringBufferInputStream - 916; 
StringReader - 923, 928; StringWriter - 
923; System.err - 941; System.in - 941; 
System.out - 941; transient - 991; typical 
I/O configurations - 927; Unicode - 923; 
write( ) - 914; writeBytes( ) . 933; 
writeChars( ) - 933; writeDouble( ) . 
934; writeExternal( ) - 986; 
writeObject( ) - 981; Writer - 914, 922, 
923; ZipEntry - 977; ZipInputStream - 
973; ZipOutputStream - 973 

Icon - 1335 

IdentityHashMap - 834, 877 

if-else statement - 116, 135 

IllegalAccessException - 573 

IllegalMonitorStateException - 1199 

Imagelcon - 1336 

immutable - 600 

implementation - 28; and interface - 257, 
316; and interface, separating - 31; and 
interface, separation - 228; hiding + 209, 
228, 352; separation of interface and 
implementation - 1321 

implements keyword - 316 

import keyword . 211 

increment operator - 101; and concurrency 
+1153 

indexed property - 1414 

indexing operator [ ] - 193 

indexOf( ), String . 592 

inference, generic type argument inference 


+632 

InflaterInputStream . 973 

inheritance - 33, 226, 237, 241, 277; and 
enum - 1020; and final - 270; and 
finalize( ) - 295; and generic code - 617; 
and synchronized - 1411; class 
inheritance diagrams - 261; combining 
composition & inheritance - 249; 
designing with inheritance - 304; 
diagram - 42; extending a class during . 
35; extending interfaces with 
inheritance - 329; from abstract classes -+ 
312; from inner classes . 382; 
initialization with inheritance . 272; 
method overloading vs. overriding - 255; 
multiple inheritance in C++ and Java - 
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326; pure inheritance vs. extension - 
306; specialization - 258; vs. 
composition - 256, 262; 830, 895 

initial capacity, of a HashMap or HashSet - 
878 

initialization: and class loading - 272; array 
initialization - 193; base class - 244; 
class - 563; class member - 239; 
constructor initialization during 
inheritance and composition - 249; 
initializing with the constructor - 155; 
instance initialization . 191, 359; lazy - 
239; member initializers - 294; non- 
static instance initialization - 191; of 
class fields - 182; of method variables - 
181; order of initialization - 185, 302; 
static - 274; with inheritance - 272 

inline method calls . 267 

inner class - 345; access rights . 348; and 
overriding - 383; and control 
frameworks - 375; and super - 383; and 
Swing - 1322; and threads - 1137; and 









upcasting - 352; anonymous inner class - 


904, 1314; and table-driven code . 859; 
callback - 372; closure - 372; generic - 
645; hidden reference to the object of 
the enclosing class - 349; identifiers and 
class files - 387; in methods & scopes - 
354; inheriting from inner classes - 382; 
local - 355; motivation - 369; nesting 
within any arbitrary scope - 355; private 
inner classes . 377; referring to the 
outer-class object - 350; static inner 
classes . 364 

InputStream - 914 

InputStreamReader . 922, 923 

instance: instance initialization - 359; non- 
static instance initialization - 191; of a 
class - 25 

instanceof . 576; and generic types - 697; 
dynamic instanceof with isInstance( ) - 
578; keyword - 569 

Integer: parselnt( ) - 1368; wrapper class - 
196 

interface: and enum - 1023; and generic 
code + 617; and implementation, 
separation of - 31, 228, 1321; and 
inheritance - 329; base-class interface + 
286; classes nested inside - 366; 
common interface - 311; for an object - 
26; initializing fields in interfaces - 335; 
keyword - 316; name collisions when 
combining interfaces - 330; nesting 
interfaces within classes and other 
interfaces - 336; private, as nested 
interfaces - 339; upcasting to an 
interface - 319; vs. abstract - 328; vs. 
implementation - 257 

internationalization, in I/O library - 923 


interrupt( ): concurrency - 1185; threading 
` 1143 

interruptible io - 1189 

Introspector - 1398 

invocation handler, for dynamic proxy + 
595 

is-a - 306; relationship, inheritance - 258; 
and upcasting - 260; vs. is-like-a 
relationships - 37 

isAssignableFrom( ), Class method - 580 

isDaemon() - 1133 

isInstance( ) - 578; and generics - 663 

isInterface( ) . 560 

is-like-a - 307 

Iterable - 629, 797; and array - 433; and 
foreach - 431 

Iterator . 406, 409, 427; hasNext( ) - 407; 
next() . 407 

Iterator design pattern - 349 
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Jacobsen, Ivar . 1457 

JApplet - 1317; menus , 1352 

JAR . 1412; file - 212; jar files and classpath 
+ 216; utility - 978 

Java: and set-top boxes - 111; AWT - 1303; 
bytecodes . 506; compiling and running 
a program . 80; Java Foundation 
Classes (JFC/Swing) . 1303; Java 
Virtual Machine (JVM) . 556; Java Web 
Start - 1376; public Java seminars - 15 

Java standard library, and thread-safety - 
1232 

JavaBeans, see Beans . 1393 

javac- 81 

javadoc - 82 

javap decompiler - 505, 610, 660 

Javassist - 1104 

JButton - 1335; Swing - 1311 

JCheckBox - 1335, 1342 

JCheckBoxMenultem . 1353, 1357 

JComboBox - 1345 

JComponent . 1337, 1360 

JDialog - 1364; menus - 1352 

JDK 1.1 I/O streams - 922 

JDK, downloading and installing - 80 

JFC, Java Foundation Classes (Swing) - 
1303 

JFileChooser - 1368 

JFrame - 1317; menus - 1352 

JIT, just-in-time compilers - 181 

JLabel - 1340 

JList - 1347 

JMenu - 1352, 1357 

JMenuBar - 1352, 1358 

JMenultem - 1336, 1352, 1357, 1358, 1360 
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JNLP, Java Network Launch Protocol - 
1376 

join( ), threading - 1143 

‘JOptionPane - 1350 

Joy, Bill - 103 

JPanel - 1334, 1360, 1392 

JPopupMenu - 1359 

JProgressBar . 1373 

JRadioButton - 1335, 1344 

JScrollPane . 1316, 1349 

JSlider - 1373 

JTabbedPane - 1349 

JTextArea «1315 

JTextField . 1312, 1338 

JTextPane . 1341 

JToggleButton - 1334 

JUnit, problems with . 1083, 

JVM (Java Virtual Machine) - 556 
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keyboard: navigation, and Swing . 1305; 
shortcuts - 1358 
keySet() - 877 
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label - 146 

labeled: break , 147; continue - 147 

late binding - 40, 277, 281 

latent typing - 721, 733 

layout, controlling layout with layout 
managers . 1317 

lazy initialization - 239 

least-recently-used (LRU) . 838 

left-shift operator (<<) . 112 

length: array member + 194; for arrays - 
749 

less than (<) - 103 

less than or equal to (<=) . 103 

lexicographic: sorting - 418; vs. alphabetic 
sorting . 783 

library: creator, vs. client programmer - 
209; design . 210; use - 210 

LIFO (last-in, first-out) + 412 

lightweight: object - 406; persistence - 980 

LineNumberInputStream - 920 

LineNumberReader - 924 

LinkedBlockingQueue . 1215 

LinkedHashMap - 834, 838, 877 

LinkedHashSet - 416, 821, 872, 874 

LinkedList - 401, 410, 423, 817 

linking; class . 563 

boxes . 1347; drop-down list - 1345 

is + 389, 394, 401, 817, 1347; 
performance comparison - 863; sorting 








and searching - 884 

listener: adapters - 1328; and events - 
1322; interfaces - 1326 

Lister, Timothy - 1459 

Listlterator - 817 

literal: class literal . 562, 576; double - 109; 
float - 109; long - 109; values - 108 

little endian - 958 

livelock - 1301 

load factor, of a HashMap or HashSet - 
878 

loader, class - 556 

loading: .class files - 214; class - 273, 563; 
initialization & class loading . 272 

local: inner class . 355; variable - 71 

lock: contention, in concurrency . 1272; 
explicit, in concurrency « 1157; in 
concurrency - 1155; optimistic locking + 
1290 

lock-free code, in concurrent 
programming - 1161 

locking, file - 970, 971 

logarithms, natural - 110 

logging, building logging into exceptions - 
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logical: AND - 120; operator and short- 
circuiting - 106; operators - 105; OR « 
120 

long: and threading - 1161; literal value 
marker (L) - 109 

look & feel, pluggable - 1373 

LRU, least-recently-used . 838 

lvalue - 95 





machines, state, and enum - 1041 

Macromedia Flex - 1416 

main() - 242 

manifest file, for JAR files - 978, 1412 

Map - 389, 394, 419; EnumMap - 1030; in- 
depth exploration of - 831; performance 
comparison - 875 

Map.Entry - 845 

MappedByteBuffer - 966 

mark() - 926 

marker annotation - 1061 

matcher, regular expression - 531 

matches( ), String - 525 

Math.random( ) - 419; range of results - 
871 

mathematical ‘operators + 98,971 

member: initializers - 294; member 
function - 29; object - 32 

memory exhaustion, solution via 
References . 890 

memory-mapped files - 966 

menu: JDialog, JApplet, JFrame . 1352; 
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JPopupMenu - 1359 

message box, in Swing - 1350 

message, sending - 27 

Messenger idiom - 621, 797, 860 

meta-annotations - 1063 

Metadata . 1059 

method: adding more methods to a design 
- 235; aliasing during method calls - 97; 
applying a method to a sequence - 728; 
behavior of polymorphic methods inside 
constructors - 301; distinguishing 
overloaded methods - 160; final - 267, 
282, 303; generic - 631; initialization of 
method variables - 181; inline method 
calls - 267; inner classes in methods & 
scopes + 354; lookup tool - 1324; method 
call binding - 281; overloading - 158; 
overriding private - 290; polymorphic 
method call - 277; private - 303; 
protected methods - 259; recursive - 
510; static . 172, 282 

Method . 1401; for reflection - 589 

MethodDescriptors - 1401 

Meyer, Jeremy . 1059, 1100, 1376 

Meyers, Scott - 30 

microbenchmarks . 871 

Microsoft Visual BASIC . 1394 

migration compatibility . 655 

missed signals, concurrency - 1203 

mistakes, and design - 234 

mixin . 713 

mkdirs( ) - 914 

mnemonics (keyboard shortcuts) - 1358 

Mock Object - 606 

modulus - 98 

monitor, for concurrency - 1155 

Mono: 58 

multicast - 1406; event, and JavaBeans - 
1407 

multidimensional arrays - 754 

multiparadigm programming - 25 

multiple dispatching: and enum - 1047; 
with EnumMap - 1055 

multiple implementation inheritance - 371 

multiple inheritance, in C++ and Java - 
326 

multiplication . 98 

multiply nested class - 368 

multitasking - 1112 

mutual exclusion (mutex), concurrency + 
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MXML, Macromedia Flex input format - 
1416 

mxmlc, Macromedia Flex compiler - 1418 





name: clash - 211; collisions - 217; 


collisions when combining interfaces + 
330; creating unique package names + 
214; qualified - 560 

namespaces , 211 

narrowing conversion + 120 

natural logarithms . 110 

nested class (static inner class) - 364 

nesting interfaces - 336 

net.mindview.util.SwingConsole - 1310 

network I/O - 946 

Neville, Sean - 1416 

new 1/0 - 946 

new operator - 173; and primitives, array - 
195 

newInstance( ) - 1335; reflection . 561 

next( ), Iterator - 407 

nio - 946; and interruption - 1189; buffer . 
946; channel - 946; performance , 967 

no-arg constructor + 156, 166 

North, BorderLayout ' 1317 

not equivalent (!=) - 103 

NOT, logical (!) . 105 

notifyAll() - 1198 

notifyListeners( ) - 1411 

null - 67 

Null Iterator design pattern . 598 

Null Object design pattern . 598 

NullPointerException - 469 

numbers, binary « 109 
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object - 25; aliasing - 97; arrays are first- 
class objects - 749; assigning objects by 
copying references - 96; Class object - 
556, 998, 1156; creation - 156; equals( ) - 
104; equivalence - 103; equivalence vs. 
reference equivalence - 104; final - 263; 
getClass( ) - 558; hashCode( ) - 833; 
interface to - 26; lock, for concurrency - 
1155; member - 32; object-oriented 
programming - 553; process of creation - 
189; serialization - 980; standard root 
class, default inheritance from - 241; 
wait( ) and notifyAll() - 1199; web of 
objects . 981 

object pool - 1246 

object-oriented, basic concepts of object- 
oriented programming (OOP) - 23 

ObjectOutputStream - 981 

Octal - 109 

ones complement operator . 111 

OOP: basic characteristics - 25; basic 
concepts of object-oriented 
programming - 23; protocol - 316; 
Simula-67 programming language . 26; 
substitutability - 25 

OpenLaszlo, alternative to Flex - 1416 
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operating system, executing programs 
from within Java - 944 

operation, atomic - 1160 

operator - 94; + and += overloading for 
String - 242; +, for String - 504; binary - 
111; bitwise - 111; casting - 120; comma 
operator - 140; common pitfalls - 119; 
indexing operator [ ] . 193; logical - 105; 
logical operators and short-circuiting - 
106; ones-complement - 111; operator 
overloading for String . 504; 
overloading - 118; precedence - 95; 
relational - 103; shift - 112; String 
conversion with operator + . 95, 118; 
ternary - 116; unary - 101, 111 

optional methods, in the Java containers - 
813 

OR - 120; (||) «105 

order: of constructor calls with inheritance 
+ 293; of initialization . 185, 272, 302 

ordinal( ), for enum - 1012 

organization, code . 221 

OSExecute . 944 

OutputStream - 914, 917 

OutputStreamWriter - 922, 923 

overflow, and primitive types - 133 

overloading: and constructors - 158; 
distinguishing overloaded methods - 
160; generics - 699; lack of name hiding, 
during inheritance . 255; method 
overloading - 158; on return values - 
165; operator + and += overloading for 
String - 242, 504; operator overloading - 
118; vs. overriding - 255 

overriding: and inner classes . 383; 
function - 36; private methods - 290; vs. 
overloading . 255 





P 


package - 210; access, and friendly - 221; 
and directory structure - 220; creating 
unique package names - 214; default - 
211, 223; names, capitalization - 75; 
package access, and protected . 258 

paintComponent( ) . 1360, 1368 

painting on a JPanel in Swing . 1360 

parameter, collecting - 713, 742 

parameterized types - 617 

parseInt( ) - 1368 

pattern, regular expression . 527 

perfect hashing function - 848 

performance: and final - 271; nio - 967; 
test, containers - 859; tuning, for 
concurrency - 1270 

persistence - 996; lightweight persistence - 
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PhantomReference - 889 


philosophers, dining, example of deadlock 
in concurrency - 1224 

pipe - 915 

piped streams - 936 

PipedInputStream - 916 

PipedOutputStream - 916, 917 

PipedReader - 923, 1221 

PipedWriter - 923, 1221 

pipes, and I/O + 1221 

Plauger, P.J. - 1458 

pluggable look & feel - 1373 

pointer, Java exclusion of pointers - 372 

polymorphism - 38, 277, 310, 554, 613; and 
constructors - 293; and multiple 
dispatching - 1048; behavior of 
polymorphic methods inside 
constructors - 301 

pool, object - 1246 

portability in C, C++ and Java - 123 

position, absolute, when laying out Swing 
components , 1320 

possessive quantifiers + 529 

post-decrement - 102 

postfix - 102 

post-increment . 102 

pre-decrement . 102 

preferences API - 1006 

prefix - 102 

pre-increment - 102 

prerequisites, for this book - 23 

primitive: comparison - 104; data types, 
and use with operators - 123; final - 263; 
final static primitives - 264; 
initialization of class fields - 182; types + 
65 

primordial class loader - 556 

printf() - 514 

printStackTrace( ) . 458, 461 

PrintStream . 921 

PrintWriter - 924, 930, 932; convenience 
constructor in Java SE5 + 937 

priority, concurrency - 1127 

PriorityBlockingQueue, for concurrency + 
1239 

PriorityQueue . 425, 827 

private - 31, 210, 221, 224, 258, 1155; 
illusion of overriding private methods - 
268; inner classes . 377; interfaces, 
when nested - 339; method overriding + 
290; methods . 303 

problem space . 24 

process control - 944 

process, concurrent - 1112 

ProcessBuilder - 944 

ProcessFiles - 1100- 

producer-consumer, concurrency - 1208 

programmer, client - 30 

programming: basic concepts of object- 
oriented programming (OOP) - 23; 
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event-driven programming - 1312; 
Extreme Programming (XP) - 1457; 
multiparadigm - 25; object-oriented - 


553 

progress bar - 1371 

promotion, to int - 122, 132 

property - 1394; bound properties - 1414; 
constrained properties - 1414; custom 
property editor - 1414; custom property 
sheet . 1414; indexed property - 1414 

PropertyChangeEvent - 1414 

PropertyDescriptors - 1400 

PropertyVetoException - 1414 

protected - 31, 210, 221, 225, 258; and 
package access - 258; is also package 
access - 227 

protocol . 316 

proxy: and java.lang.ref.Reference - 890; 
for unmodifiable methods in the 
Collections class - 817 

Proxy design pattern - 593 

public - 31, 210, 221, 222; and interface - 
316; class, and compilation units - 211 

pure substitution - 37, 307 

PushbackInputStream - 920 

PushbackReader . 924 

pushdown stack - 412; generic - 625 

Python - 1, 5, 9, 53, 60, 722, 787, 1113, 
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